Coalesce multiple RwLock
s into one LanguageRegistryState
struct
This commit is contained in:
parent
e30ea43a14
commit
a8ac08f5bd
1 changed files with 52 additions and 43 deletions
|
@ -475,8 +475,7 @@ struct AvailableLanguage {
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct LanguageRegistry {
|
pub struct LanguageRegistry {
|
||||||
languages: RwLock<Vec<Arc<Language>>>,
|
state: RwLock<LanguageRegistryState>,
|
||||||
available_languages: RwLock<Vec<AvailableLanguage>>,
|
|
||||||
language_server_download_dir: Option<Arc<Path>>,
|
language_server_download_dir: Option<Arc<Path>>,
|
||||||
lsp_binary_statuses_tx: async_broadcast::Sender<(Arc<Language>, LanguageServerBinaryStatus)>,
|
lsp_binary_statuses_tx: async_broadcast::Sender<(Arc<Language>, LanguageServerBinaryStatus)>,
|
||||||
lsp_binary_statuses_rx: async_broadcast::Receiver<(Arc<Language>, LanguageServerBinaryStatus)>,
|
lsp_binary_statuses_rx: async_broadcast::Receiver<(Arc<Language>, LanguageServerBinaryStatus)>,
|
||||||
|
@ -488,26 +487,33 @@ pub struct LanguageRegistry {
|
||||||
Shared<BoxFuture<'static, Result<PathBuf, Arc<anyhow::Error>>>>,
|
Shared<BoxFuture<'static, Result<PathBuf, Arc<anyhow::Error>>>>,
|
||||||
>,
|
>,
|
||||||
>,
|
>,
|
||||||
subscription: RwLock<(watch::Sender<()>, watch::Receiver<()>)>,
|
|
||||||
theme: RwLock<Option<Arc<Theme>>>,
|
|
||||||
executor: Option<Arc<Background>>,
|
executor: Option<Arc<Background>>,
|
||||||
version: AtomicUsize,
|
}
|
||||||
|
|
||||||
|
struct LanguageRegistryState {
|
||||||
|
languages: Vec<Arc<Language>>,
|
||||||
|
available_languages: Vec<AvailableLanguage>,
|
||||||
|
subscription: (watch::Sender<()>, watch::Receiver<()>),
|
||||||
|
theme: Option<Arc<Theme>>,
|
||||||
|
version: usize,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl LanguageRegistry {
|
impl LanguageRegistry {
|
||||||
pub fn new(login_shell_env_loaded: Task<()>) -> Self {
|
pub fn new(login_shell_env_loaded: Task<()>) -> Self {
|
||||||
let (lsp_binary_statuses_tx, lsp_binary_statuses_rx) = async_broadcast::broadcast(16);
|
let (lsp_binary_statuses_tx, lsp_binary_statuses_rx) = async_broadcast::broadcast(16);
|
||||||
Self {
|
Self {
|
||||||
|
state: RwLock::new(LanguageRegistryState {
|
||||||
|
languages: vec![PLAIN_TEXT.clone()],
|
||||||
|
available_languages: Default::default(),
|
||||||
|
subscription: watch::channel(),
|
||||||
|
theme: Default::default(),
|
||||||
|
version: 0,
|
||||||
|
}),
|
||||||
language_server_download_dir: None,
|
language_server_download_dir: None,
|
||||||
languages: RwLock::new(vec![PLAIN_TEXT.clone()]),
|
|
||||||
available_languages: Default::default(),
|
|
||||||
lsp_binary_statuses_tx,
|
lsp_binary_statuses_tx,
|
||||||
lsp_binary_statuses_rx,
|
lsp_binary_statuses_rx,
|
||||||
login_shell_env_loaded: login_shell_env_loaded.shared(),
|
login_shell_env_loaded: login_shell_env_loaded.shared(),
|
||||||
lsp_binary_paths: Default::default(),
|
lsp_binary_paths: Default::default(),
|
||||||
subscription: RwLock::new(watch::channel()),
|
|
||||||
theme: Default::default(),
|
|
||||||
version: Default::default(),
|
|
||||||
executor: None,
|
executor: None,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -529,42 +535,41 @@ impl LanguageRegistry {
|
||||||
lsp_adapter: Option<Box<dyn LspAdapter>>,
|
lsp_adapter: Option<Box<dyn LspAdapter>>,
|
||||||
get_queries: fn(&str) -> LanguageQueries,
|
get_queries: fn(&str) -> LanguageQueries,
|
||||||
) {
|
) {
|
||||||
self.available_languages.write().push(AvailableLanguage {
|
self.state
|
||||||
path,
|
.write()
|
||||||
config,
|
.available_languages
|
||||||
grammar,
|
.push(AvailableLanguage {
|
||||||
lsp_adapter,
|
path,
|
||||||
get_queries,
|
config,
|
||||||
});
|
grammar,
|
||||||
|
lsp_adapter,
|
||||||
|
get_queries,
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn language_names(&self) -> Vec<String> {
|
pub fn language_names(&self) -> Vec<String> {
|
||||||
let mut result = self
|
let state = self.state.read();
|
||||||
|
let mut result = state
|
||||||
.available_languages
|
.available_languages
|
||||||
.read()
|
|
||||||
.iter()
|
.iter()
|
||||||
.map(|l| l.config.name.to_string())
|
.map(|l| l.config.name.to_string())
|
||||||
.chain(
|
.chain(state.languages.iter().map(|l| l.config.name.to_string()))
|
||||||
self.languages
|
|
||||||
.read()
|
|
||||||
.iter()
|
|
||||||
.map(|l| l.config.name.to_string()),
|
|
||||||
)
|
|
||||||
.collect::<Vec<_>>();
|
.collect::<Vec<_>>();
|
||||||
result.sort_unstable_by_key(|language_name| language_name.to_lowercase());
|
result.sort_unstable_by_key(|language_name| language_name.to_lowercase());
|
||||||
result
|
result
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn workspace_configuration(&self, cx: &mut MutableAppContext) -> Task<serde_json::Value> {
|
pub fn workspace_configuration(&self, cx: &mut MutableAppContext) -> Task<serde_json::Value> {
|
||||||
|
let state = self.state.read();
|
||||||
let mut language_configs = Vec::new();
|
let mut language_configs = Vec::new();
|
||||||
for language in self.available_languages.read().iter() {
|
for language in &state.available_languages {
|
||||||
if let Some(adapter) = language.lsp_adapter.as_ref() {
|
if let Some(adapter) = language.lsp_adapter.as_ref() {
|
||||||
if let Some(language_config) = adapter.workspace_configuration(cx) {
|
if let Some(language_config) = adapter.workspace_configuration(cx) {
|
||||||
language_configs.push(language_config);
|
language_configs.push(language_config);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
for language in self.languages.read().iter() {
|
for language in &state.languages {
|
||||||
if let Some(adapter) = language.lsp_adapter() {
|
if let Some(adapter) = language.lsp_adapter() {
|
||||||
if let Some(language_config) = adapter.workspace_configuration(cx) {
|
if let Some(language_config) = adapter.workspace_configuration(cx) {
|
||||||
language_configs.push(language_config);
|
language_configs.push(language_config);
|
||||||
|
@ -583,25 +588,27 @@ impl LanguageRegistry {
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn add(&self, language: Arc<Language>) {
|
pub fn add(&self, language: Arc<Language>) {
|
||||||
if let Some(theme) = self.theme.read().clone() {
|
let mut state = self.state.write();
|
||||||
|
if let Some(theme) = state.theme.as_ref() {
|
||||||
language.set_theme(&theme.editor.syntax);
|
language.set_theme(&theme.editor.syntax);
|
||||||
}
|
}
|
||||||
self.languages.write().push(language);
|
state.languages.push(language);
|
||||||
self.version.fetch_add(1, SeqCst);
|
state.version += 1;
|
||||||
*self.subscription.write().0.borrow_mut() = ();
|
*state.subscription.0.borrow_mut() = ();
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn subscribe(&self) -> watch::Receiver<()> {
|
pub fn subscribe(&self) -> watch::Receiver<()> {
|
||||||
self.subscription.read().1.clone()
|
self.state.read().subscription.1.clone()
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn version(&self) -> usize {
|
pub fn version(&self) -> usize {
|
||||||
self.version.load(SeqCst)
|
self.state.read().version
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn set_theme(&self, theme: Arc<Theme>) {
|
pub fn set_theme(&self, theme: Arc<Theme>) {
|
||||||
*self.theme.write() = Some(theme.clone());
|
let mut state = self.state.write();
|
||||||
for language in self.languages.read().iter() {
|
state.theme = Some(theme.clone());
|
||||||
|
for language in &state.languages {
|
||||||
language.set_theme(&theme.editor.syntax);
|
language.set_theme(&theme.editor.syntax);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -654,19 +661,21 @@ impl LanguageRegistry {
|
||||||
) -> UnwrapFuture<oneshot::Receiver<Result<Arc<Language>>>> {
|
) -> UnwrapFuture<oneshot::Receiver<Result<Arc<Language>>>> {
|
||||||
let (tx, rx) = oneshot::channel();
|
let (tx, rx) = oneshot::channel();
|
||||||
|
|
||||||
if let Some(language) = self
|
let mut state = self.state.write();
|
||||||
|
if let Some(language) = state
|
||||||
.languages
|
.languages
|
||||||
.read()
|
|
||||||
.iter()
|
.iter()
|
||||||
.find(|language| callback(&language.config))
|
.find(|language| callback(&language.config))
|
||||||
{
|
{
|
||||||
let _ = tx.send(Ok(language.clone()));
|
let _ = tx.send(Ok(language.clone()));
|
||||||
} else if let Some(executor) = self.executor.clone() {
|
} else if let Some(executor) = self.executor.clone() {
|
||||||
let mut available_languages = self.available_languages.write();
|
if let Some(ix) = state
|
||||||
|
.available_languages
|
||||||
if let Some(ix) = available_languages.iter().position(|l| callback(&l.config)) {
|
.iter()
|
||||||
let language = available_languages.remove(ix);
|
.position(|l| callback(&l.config))
|
||||||
drop(available_languages);
|
{
|
||||||
|
let language = state.available_languages.remove(ix);
|
||||||
|
drop(state);
|
||||||
let name = language.config.name.clone();
|
let name = language.config.name.clone();
|
||||||
let this = self.clone();
|
let this = self.clone();
|
||||||
executor
|
executor
|
||||||
|
@ -702,7 +711,7 @@ impl LanguageRegistry {
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn to_vec(&self) -> Vec<Arc<Language>> {
|
pub fn to_vec(&self) -> Vec<Arc<Language>> {
|
||||||
self.languages.read().iter().cloned().collect()
|
self.state.read().languages.iter().cloned().collect()
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn start_language_server(
|
pub fn start_language_server(
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue