aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorJoe Wilm <joe@jwilm.com>2016-12-29 15:30:30 -0500
committerJoe Wilm <joe@jwilm.com>2016-12-29 15:34:33 -0500
commitd06360216d3f10a0d42f576e6a859de88436a7d6 (patch)
tree0799bf3654f3b4caa816102ae6985cdf51794321
parent689e0f4ad784390c39ed614358f14a722be7dc17 (diff)
downloadalacritty-d06360216d3f10a0d42f576e6a859de88436a7d6.tar.gz
alacritty-d06360216d3f10a0d42f576e6a859de88436a7d6.zip
Hopefully fix read not triggering draw
The terminal mutex is no longer released between event processing and doing a draw. This may fix the race condition with data arriving but not being displayed until a subsequent event. cc #29
-rw-r--r--src/event.rs108
-rw-r--r--src/main.rs5
2 files changed, 57 insertions, 56 deletions
diff --git a/src/event.rs b/src/event.rs
index a231e898..47621af3 100644
--- a/src/event.rs
+++ b/src/event.rs
@@ -2,14 +2,15 @@
use std::borrow::Cow;
use std::fs::File;
use std::io::Write;
-use std::sync::{Arc, mpsc};
-use serde_json as json;
+use std::sync::mpsc;
+use serde_json as json;
+use parking_lot::MutexGuard;
use glutin::{self, ElementState};
-use index::{Line, Column, Side};
use config::Config;
use display::OnResize;
+use index::{Line, Column, Side};
use input::{self, ActionContext, MouseBinding, KeyBinding};
use selection::Selection;
use sync::FairMutex;
@@ -58,7 +59,6 @@ pub struct Processor<N> {
mouse_bindings: Vec<MouseBinding>,
notifier: N,
mouse: Mouse,
- terminal: Arc<FairMutex<Term>>,
resize_tx: mpsc::Sender<(u32, u32)>,
ref_test: bool,
size_info: SizeInfo,
@@ -81,21 +81,15 @@ impl<N: Notify> Processor<N> {
/// pty.
pub fn new(
notifier: N,
- terminal: Arc<FairMutex<Term>>,
resize_tx: mpsc::Sender<(u32, u32)>,
config: &Config,
ref_test: bool,
+ size_info: SizeInfo,
) -> Processor<N> {
- let size_info = {
- let terminal = terminal.lock();
- terminal.size_info().to_owned()
- };
-
Processor {
key_bindings: config.key_bindings().to_vec(),
mouse_bindings: config.mouse_bindings().to_vec(),
notifier: notifier,
- terminal: terminal,
resize_tx: resize_tx,
ref_test: ref_test,
mouse: Default::default(),
@@ -176,57 +170,65 @@ impl<N: Notify> Processor<N> {
}
/// Process at least one event and handle any additional queued events.
- pub fn process_events(&mut self, window: &Window) -> bool {
+ pub fn process_events<'a>(
+ &mut self,
+ term: &'a FairMutex<Term>,
+ window: &Window
+ ) -> (MutexGuard<'a, Term>, bool) {
let mut wakeup_request = false;
- // These are lazily initialized the first time an event is returned
- // from the blocking WaitEventsIterator. Otherwise, the pty reader
- // would be blocked the entire time we wait for input!
+ // Terminal is lazily initialized the first time an event is returned
+ // from the blocking WaitEventsIterator. Otherwise, the pty reader would
+ // be blocked the entire time we wait for input!
let terminal;
- let context;
- let mut processor: input::Processor<N>;
-
- // Convenience macro which curries most arguments to handle_event.
- macro_rules! process {
- ($event:expr) => {
- Processor::handle_event(
- &mut processor,
- $event,
- &mut wakeup_request,
- self.ref_test,
- &self.resize_tx,
- )
+
+ {
+ // Ditto on lazy initialization for context and processor.
+ let context;
+ let mut processor: input::Processor<N>;
+
+ // Convenience macro which curries most arguments to handle_event.
+ macro_rules! process {
+ ($event:expr) => {
+ Processor::handle_event(
+ &mut processor,
+ $event,
+ &mut wakeup_request,
+ self.ref_test,
+ &self.resize_tx,
+ )
+ }
}
- }
- match window.wait_events().next() {
- Some(event) => {
- terminal = self.terminal.lock();
- context = ActionContext {
- terminal: &terminal,
- notifier: &mut self.notifier,
- selection: &mut self.selection,
- mouse: &mut self.mouse,
- size_info: &self.size_info,
- };
-
- processor = input::Processor {
- ctx: context,
- key_bindings: &self.key_bindings[..],
- mouse_bindings: &self.mouse_bindings[..]
- };
+ match window.wait_events().next() {
+ Some(event) => {
+ terminal = term.lock();
+ context = ActionContext {
+ terminal: &terminal,
+ notifier: &mut self.notifier,
+ selection: &mut self.selection,
+ mouse: &mut self.mouse,
+ size_info: &self.size_info,
+ };
+
+ processor = input::Processor {
+ ctx: context,
+ key_bindings: &self.key_bindings[..],
+ mouse_bindings: &self.mouse_bindings[..]
+ };
+
+ process!(event);
+ },
+ // Glutin guarantees the WaitEventsIterator never returns None.
+ None => unreachable!(),
+ }
+ for event in window.poll_events() {
process!(event);
- },
- // Glutin guarantees the WaitEventsIterator never returns None.
- None => unreachable!(),
- }
-
- for event in window.poll_events() {
- process!(event);
+ }
}
- wakeup_request
+ (terminal, wakeup_request)
}
pub fn update_config(&mut self, config: &Config) {
diff --git a/src/main.rs b/src/main.rs
index fb58b461..aa2da104 100644
--- a/src/main.rs
+++ b/src/main.rs
@@ -104,10 +104,10 @@ fn run(mut config: Config, options: cli::Options) -> Result<(), Box<Error>> {
// Need the Rc<RefCell<_>> here since a ref is shared in the resize callback
let mut processor = event::Processor::new(
event_loop::Notifier(loop_tx),
- terminal.clone(),
display.resize_channel(),
&config,
options.ref_test,
+ display.size().to_owned(),
);
// Create a config monitor when config was loaded from path
@@ -123,7 +123,7 @@ fn run(mut config: Config, options: cli::Options) -> Result<(), Box<Error>> {
// Main display loop
loop {
// Process input and window events
- let wakeup_request = processor.process_events(display.window());
+ let (mut terminal, wakeup_request) = processor.process_events(&terminal, display.window());
// Handle config reloads
let config_updated = config_monitor.as_ref()
@@ -136,7 +136,6 @@ fn run(mut config: Config, options: cli::Options) -> Result<(), Box<Error>> {
}).unwrap_or(false);
// Maybe draw the terminal
- let mut terminal = terminal.lock();
if wakeup_request || config_updated {
// Handle pending resize events
//