Skip to content

Commit

Permalink
restore: restrict last word to valid candidates also for 12/18 words
Browse files Browse the repository at this point in the history
So far we've only been doing this for 24 words, where there are 8 list
of valid candidate, so they could be shown in a menu.

We do the same for 12/18 words, but since there are many more
candidates (128 for 12 words, 32 for 18 words), we restrict the
trinary keyboard entry to this list instead of showing them all in a
menu.

This allows easily importing seeds that were made without computers by
rolling dices or similar, helping with the checksum issue, like we do
now for 24 word mnemonics.

One downside is that if the user has a typo in the previous words,
they will be unable to enter the last word as it will be missing from
the candidates, which could be confusing. With 24 words, there is an
explicit "None of them" entry to tell the user that their words are
invalid, but that only works in the menu. With the 12th/18th word
still using the keyboard, there is no good way of doing this.
  • Loading branch information
benma committed Nov 3, 2023
1 parent 1fc4f9f commit 0435d2b
Show file tree
Hide file tree
Showing 2 changed files with 89 additions and 33 deletions.
3 changes: 3 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,9 @@ customers cannot upgrade their bootloader, its changes are recorded separately.
- Disable screensaver when displaying a receive address, confirming a transaction, and other interactive actions
- Add Sepolia testnet for Ethereum
- Added a disclaimer screen before the recovery words are displayed
- Verifiable seed generation: when restoring from 12 or 18 recovery words, for the final word, restrict input to the valid candidate words which result in a valid checksum.
For 24 words, this feature was already introduced in 9.4.0.


### 9.15.0
- Security bugfix: check index of an input's previous output to prevent the fee attack originally
Expand Down
119 changes: 86 additions & 33 deletions src/rust/bitbox02-rust/src/workflow/mnemonic.rs
Original file line number Diff line number Diff line change
Expand Up @@ -69,7 +69,8 @@ pub async fn confirm_word(choices: &[&str], title: &str) -> Result<u8, CancelErr
/// such that the resulting bip39 phrase has a valid checksum. There are always exactly 8 such words
/// for 24 word mnemonics, 32 words for 18 word mnemonics and 128 words for 12 word mnemonics.
/// `entered_words` must contain 11/17/23 words from the BIP39 wordlist.
fn lastword_choices(entered_words: &[&str]) -> Vec<zeroize::Zeroizing<String>> {
/// The result is the list of indices of the words in the BIP39 wordlist.
fn lastword_choices(entered_words: &[&str]) -> Vec<u16> {
let (seed_len_bits, checksum_len_bits, bitmask_seed) = match entered_words.len() {
11 => (128, 4, 0b10000000),
17 => (192, 6, 0b11100000),
Expand Down Expand Up @@ -116,11 +117,18 @@ fn lastword_choices(entered_words: &[&str]) -> Vec<zeroize::Zeroizing<String>> {
// Last word is 11 bits: <last 7/5/3 bits of the seed || 4/6/8 bits checksum>.
let word_idx: u16 =
(i << checksum_len_bits) | (hash[0] >> (8 - checksum_len_bits)) as u16;
bitbox02::keystore::get_bip39_word(word_idx).unwrap()
word_idx
})
.collect()
}

fn lastword_choices_strings(entered_words: &[&str]) -> Vec<zeroize::Zeroizing<String>> {
lastword_choices(entered_words)
.into_iter()
.map(|word_idx| bitbox02::keystore::get_bip39_word(word_idx).unwrap())
.collect()
}

/// Select the 24th word from a list of 8 valid candidate words presented as a menu.
/// Returns `Ok(None)` if the user chooses "None of them".
/// Returns `Ok(Some(word))` if the user chooses a word.
Expand All @@ -129,7 +137,7 @@ async fn get_24th_word(
title: &str,
entered_words: &[&str],
) -> Result<Option<zeroize::Zeroizing<String>>, CancelError> {
let mut choices = lastword_choices(entered_words);
let mut choices = lastword_choices_strings(entered_words);
// Add one more menu entry.
let none_of_them_idx = {
choices.push(zeroize::Zeroizing::new("None of them".into()));
Expand Down Expand Up @@ -166,6 +174,46 @@ async fn get_24th_word(
}
}

/// Select the last word of a 12 or 18 word mnemonic from a list of valid candidate words. The input
/// is the trinary input keyboard with the wordlist restricted to these candidates.
///
/// Returns `Ok(word)` if the user chooses a word.
/// Returns `Err(CancelError::Cancelled)` if the user cancels.
async fn get_12th_18th_word(
title: &str,
entered_words: &[&str],
) -> Result<zeroize::Zeroizing<String>, CancelError> {
// With 12/18 words there are 128/32 candidates, so we limit the keyboard to allow entering only
// these.
loop {
let choices = lastword_choices(entered_words);
let candidates = bitbox02::keystore::get_bip39_wordlist(Some(&choices));
let word = trinary_input_string::enter(
&trinary_input_string::Params {
title,
wordlist: Some(&candidates),
..Default::default()
},
trinary_input_string::CanCancel::Yes,
"",
)
.await
.map(|s| s.as_string())?;

// Confirm word picked again, as a typo here would be extremely annoying. Double checking
// is also safer, as the user might not even realize they made a typo.
if let Ok(()) = confirm::confirm(&confirm::Params {
title,
body: &word,
..Default::default()
})
.await
{
return Ok(word);
}
}
}

/// Retrieve a BIP39 mnemonic sentence of 12, 18 or 24 words from the user.
pub async fn get() -> Result<zeroize::Zeroizing<String>, CancelError> {
let num_words: usize = match choose("How many words?", "12", "18", "24").await {
Expand Down Expand Up @@ -194,30 +242,35 @@ pub async fn get() -> Result<zeroize::Zeroizing<String>, CancelError> {
// goes forward again.
let preset = entered_words[word_idx].as_str();

let user_entry: Result<zeroize::Zeroizing<String>, CancelError> = if word_idx == 23 {
// For the last word, we can restrict to a subset of bip39 words that fulfil the
// checksum requirement. We do this only when entering 24 words, which results in a
// small list of 8 valid candidates. This special case exists so that users can
// generate a seed using only the device and no external software, allowing seed
// generation via dice throws, for example.
match get_24th_word(&title, &as_str_vec(&entered_words[..word_idx])).await {
Ok(None) => return Err(CancelError::Cancelled),
Ok(Some(r)) => Ok(r),
Err(e) => Err(e),
}
} else {
trinary_input_string::enter(
&trinary_input_string::Params {
title: &title,
wordlist: Some(&bip39_wordlist),
..Default::default()
},
trinary_input_string::CanCancel::Yes,
preset,
)
.await
.map(|s| s.as_string())
};
let user_entry: Result<zeroize::Zeroizing<String>, CancelError> =
if word_idx == num_words - 1 {
// For the last word, we can restrict to a subset of bip39 words that fulfil the
// checksum requirement. This special case exists so that users can generate a seed
// using only the device and no external software, allowing seed generation via dice
// throws, for example.
if num_words == 24 {
// With 24 words there are only 8 valid candidates. We presnet them as a menu.
match get_24th_word(&title, &as_str_vec(&entered_words[..word_idx])).await {
Ok(None) => return Err(CancelError::Cancelled),
Ok(Some(r)) => Ok(r),
Err(e) => Err(e),
}
} else {
get_12th_18th_word(&title, &as_str_vec(&entered_words[..word_idx])).await
}
} else {
trinary_input_string::enter(
&trinary_input_string::Params {
title: &title,
wordlist: Some(&bip39_wordlist),
..Default::default()
},
trinary_input_string::CanCancel::Yes,
preset,
)
.await
.map(|s| s.as_string())
};

match user_entry {
Err(CancelError::Cancelled) => {
Expand Down Expand Up @@ -305,13 +358,13 @@ mod tests {
);

assert_eq!(
&lastword_choices(&["violin"; 23]),
&lastword_choices_strings(&["violin"; 23]),
&bruteforce_lastword(&["violin"; 23]),
);

let mnemonic = "side stuff card razor rescue enhance risk exchange ozone render large describe gas juice offer permit vendor custom forget lecture divide junior narrow".split(' ').collect::<Vec<&str>>();
assert_eq!(
&lastword_choices(&mnemonic),
&lastword_choices_strings(&mnemonic),
&bruteforce_lastword(&mnemonic)
);

Expand All @@ -328,13 +381,13 @@ mod tests {
);

assert_eq!(
&lastword_choices(&["violin"; 17]),
&lastword_choices_strings(&["violin"; 17]),
&bruteforce_lastword(&["violin"; 17]),
);

let mnemonic = "alpha write diary chicken cable spoil dirt hair bike fiction system bright mimic garage giggle involve leisure".split(' ').collect::<Vec<&str>>();
assert_eq!(
&lastword_choices(&mnemonic),
&lastword_choices_strings(&mnemonic),
&bruteforce_lastword(&mnemonic)
);

Expand Down Expand Up @@ -362,15 +415,15 @@ mod tests {
);

assert_eq!(
&lastword_choices(&["violin"; 11]),
&lastword_choices_strings(&["violin"; 11]),
&bruteforce_lastword(&["violin"; 11]),
);

let mnemonic = "outer elite desert faint cliff useless teach screen combine exercise below"
.split(' ')
.collect::<Vec<&str>>();
assert_eq!(
&lastword_choices(&mnemonic),
&lastword_choices_strings(&mnemonic),
&bruteforce_lastword(&mnemonic)
);
}
Expand Down

0 comments on commit 0435d2b

Please sign in to comment.