cargo install --git https://github.com/MathiasPius/crates-lsp.git --rev 53fb3b8d58c43ea799fdbc1fd28fa5db3f01b01a # ensure that ~/.cargo/bin is in $PATH
Add language definition to your helix languages file at ~/.config/helix/languages.toml (create if not there)
Copy runtime/queries/toml to your config folder as queries
1 2 3
helix --health # find out where your runtime directory and config directories are mkdir -p ~/.config/helix/runtime/queries # create the subfolder, substitute path to yours cp -r /usr/lib/helix/runtime/queries/toml ~/.config/helix/runtime/queries/crates # copy the queries definition, substitute paths to yours
Why the queries copy?
It seems that at this time, language definitions are not able to point to other queries (formatting setups). Copying the runtime/queries subfolder is the easiest solution, thanks djo_kirawi at #helix.
Coming from VSCode
I’ve been a VSCode user for ages. Vim has always both fascinated and scared me, but I was never able to make the jump. I think similar to many others I’ve found the plugins and configuration just too much work to get me where I’d like my editor to be.
I found out about helix two weeks ago and really liked it. The select-action model resonated with me much better, since I already used it in my texel ascii art editor although I didn’t realize at the time that it has a name, or truly that it was even different from vim’s model.
Most things in helix just worked for me but the one major missing ingedient was seeing dependency version in Cargo.toml files. As it turns out, it was much easier to add than I expected :)
SPI is designed such that a single bus can be shared accross multiple peripheral devices. Each device requires a chip select line to tell it when to listen and reply to data.
The following diagram explains it nicely:
Embedded-HAL
The currently released version of embedded-hal@v0.2.7 does not support sharing the SPI bus, at least not explicitly. There’s no indication of how chip select is to be used or what to do with multiple devices.
While the current abstraction does not prevent shared use, it encourages HAL implementors to just use the entire bus as a single device.
The upcoming stable version of embedded-hal of embedded-hal changes the definition o that there’s a SpiBus and one or more SpiDevice. This forces the implementors into enabling sharing.
The rest of this article is about explaining the + '_ part, why it’s needed and what it solves. Also, VSCode + RLS currently does not highlight this properly, which tells me this is a pretty niche feature.
Before I begin explaining what the issue was and how it’s solved, I think it’s best to show the overall problem I was trying to “code my way around”.
The Problem Overview
I’ve been trying to refactor the code in Texel ASCII Art Editor so that I can abstract away the dependency on Termion. The main goal was to add support for Crossterm and thus enable Windows compilation. Termion and Crossterm are both “terminal libraries” allowing to both read input (keys etc.) and output text and commands to the terminal.
I already abstracted the drawing parts but still had to handle the input events loop which looked something like this:
1 2 3 4 5 6 7 8 9 10 11 12
// adds termion event parsing to Stdin use termion::input::TermRead;
// construct event mapper from a character map stored on disk letinput_map = InputMap::from(config.char_map);
// c contains termion specific Result<termion::event::Event, std::io::Error> forcinstdin().events() { // maps termion events to internal Texel events (e.g. 'x' to 'SaveAndQuit') letmapped = input_map.map_input(c.unwrap()); // do stuff with internal event }
I needed to abstract away this whole loop and mapping. I thought if I do it all in one go and refactor the existing input_map code into something like a InputSource generic struct I could easily swap implementations around.
The idea was to have a loop that looked like this:
1 2 3 4 5 6
// construct input source + mapper from a character map stored on disk letinput_source = InputSource::from(config.char_map);
formappedin input_source.events() { // do stuff with internal event }
The Code
The idea seemed straightforward at first. I’ve put the original InputMap code into the newly created Inputsource such that:
1 2 3 4 5 6
// Event is the internal Texel Event I want, e.g. `SaveAndQuit` typeRawMap = HashMap<termion::event::Event, Event>;
pubstructInputSource { map: RawMap, // to be used by the new iterator for mapping TEvent -> Event }
Now I needed to add an iterator that can consume the original termion::input::Events iterator that’s returned by the “expanded” Stdin::events() call.
This new iterator needs to unwrap the contents, unwrap the Result as well and then perform the original InputMap::map_input call somehow. Definition is:
structMappedIter<'a> { source: termion::input::Events<Stdin>, // termion iterator map: &'a RawMap, // reference to the hash map, lives at least as long as this iterator }
fnnext(&mutself) ->Option<Event> { // unwrap termion's Result and map to internal Event matchself.source.next() { None => None, Some(result) => Some(self.map_input(result.unwrap())), } } }
So far so good. I have a working “iterator-mapper” constructed with a basic ownership model. InputSource gets constructed with the required mapping from the disk (config file) and stores it as a HashMap.
A note about the first use of '_: it means anonymous lifetime and in my mind translates roughly to “explicit lifetime, but elided”, so that we don’t need to name it and there’s no need to declare it at the impl level.
In both our implementation blocks here we don’t really use the lifetime for anything and so it can just be “elided explicitly” here.
The only remaining part is the InputSource::events method which should create a new mapping iterator each round. Here’s how that looked in my first iteration:
First thing to notice here is that we use the anonymous lifetime '_ again, but in the return type. This has a different meaning than in the implementation blocks before. Whereas before it meant we don’t need an explicit named lifetime since we don’t “use” it, in here it means “single lifetime for output”. We’re basically saying that the return value is borrowing from self and thus needs to outlive this InputSource.
This code worked as-is but it has one major flaw. The MappedIter type is returned directly mandating it to be public. MappedIter is however a termion specific piece of code that I’d like to keep hidden. There’s no reason why I need the main program to “see” it either since all I care about is that it’s an Iterator<Item = Event>.
I thought I can easily solve this by changing the definition to:
The basic typing is correct, MappedIter definitely implements Iterator<Item = Event> correctly. The error also obviously points to a lifetime issue.
Having a second look made me realize that the real problem is my code is “lossy” here. If MappedIter didn’t require an explicit lifetime definition this would “just work”.
But MappedIter<'a> does require a lifetime definition because it has a reference to a HashMap inside it. The ownership model is very basic, InputSource owns the HashMap and MappedIter therefore has to live in its lifetime.
The problem is that impl Iterator<Item = Event> does not specify a lifetime, and also has no lifetime in its own definition.
At this point I thought that using the impl keyword here was impossible. How can I specify the lifetime requirement to the return type if the return type itself is a foreign trait? It’s not like I can just add <'a> to Iterator…
The Solution
The solution as mentioned at the start, is to add the odd looking + '_ lifetime “addition” at the end. I actually went to IRC channel and bothered a human about this, but if I knew how to read properly the Rust compiler already told me a few lines below the error:
1 2 3
help: you can add a constraint to the return type to make it last less than `'static` and match the anonymous lifetime #1 defined on the method body at 49:5 | 49 | pub fn events(&self) -> impl Iterator<Item = Event> + '_ {
And there we have it right? Well, sure it compiles but.. why does adding + '_ fix this and more importantly, what does it mean?
Looking at the anonymous lifetime definition explains the core lifetime problem and to an extend the solution, but the examples shown there don’t do any kind of “summing”, so what’s going on here?
In order to understand why a plus sign is used here we need to look more at what the impl keyword means in the return type definition.
The documentation clearly says it, the syntax is fn name() -> impl Trait. This means that anything coming after the impl keyword in this context is a Trait definitions. The plus sign suddenly makes sense.
Traits can be “combined” with the plus sign, and more importantly for our use case, can be also combined with lifetime definition. The whole expression becomes in essence a new trait.
Adding the + '_ just means we’re defining out Iterator trait to have an explicit lifetime that gets elided to mean “for the duration of self“ thus fixing the “lossy” problem.
As an interesting side note, VSCode does not highlight this type of syntax properly at this time, which tells me this is a pretty niche syntax.
This driver supports the MAX7219 and MAX7221 chips used in segment and led matrix displays.
I’ve tested the driver with my hifive1-revB board with two sergmented displays connected in series.
Connecting displays in series
When connecting displays in series the setup uses a non-standard SPI data send trick to avoid needing additional CS cables.
You use the DOUT pin from first display and connect it directly to the DIN (MOSI) pin on the next display.
Using the driver
To use the driver you need to instantiate embedded-hal for your specific board. Usage is fairly simple after that, here’s a short example using GPIO pins.
// init the display driver with prepared pins letmut display = MAX7219::from_pins(1, data, cs, sck).unwrap();
// make sure to wake the display up display.power_on().unwrap(); // write given octet of ASCII characters with dots specified by 3rd param bits display.write_str(0, b"pls help", 0b00100000).unwrap(); // set display intensity lower display.set_intensity(0, 0x1).unwrap();
loop {} }
I’ve created a project with examples using the driver via SPI as well as individual GPIO pins utilizing the hifive1-revB development board. You can also check out the documentation for the driver.