Skip to content

Commit e26863a

Browse files
committed
Merge #71: Add "alloc" feature
85caa28 Add CI test.sh script (Tobin C. Harding) 1ceb828 Add no-allocator crate (Tobin C. Harding) 8f27962 Add "alloc" feature (Tobin C. Harding) 33bad4c Move embedded to sub-dir with-allocator (Tobin C. Harding) Pull request description: We would like users to be able to use parts of this library in a `no_std` environment without an allocator. To achieve this add an "alloc" feature and feature gate any code that requires allocation behind. To test the new feature gating works, and catch regressions, add a crate that uses `bech32` as a dependency with no default features and forces a `no_std` build. (Does not test any code just imports the lib to verify it builds.) Patch 3 can be moved before patch 2 to verify that the crate does not currently build without an allocator. Patch 4 adds a minimal `contrib/test.sh` script. ACKs for top commit: apoelstra: ACK 85caa28 Tree-SHA512: ee1507d08dca79ddee07e6b7188c8f707d0c0dd1aaf778f3d5b376fa5ba99358260ed034efce852366e794843737757589564cbf65fa3e2560ff73559baa06d2
2 parents 8f83401 + 85caa28 commit e26863a

File tree

11 files changed

+141
-18
lines changed

11 files changed

+141
-18
lines changed

.github/workflows/rust.yml

+25-7
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,7 @@ on: [pull_request]
33
name: Continuous Integration
44

55
jobs:
6-
test:
6+
Test:
77
name: Test Suite
88
runs-on: ubuntu-latest
99
strategy:
@@ -13,16 +13,17 @@ jobs:
1313
- stable
1414
- nightly
1515
steps:
16+
- name: Checkout Crate
1617
- uses: actions/checkout@v2
18+
- name: Checkout Toolchain
1719
- uses: actions-rs/toolchain@v1
1820
with:
1921
profile: minimal
2022
toolchain: ${{ matrix.rust }}
2123
override: true
22-
- uses: actions-rs/cargo@v1
23-
with:
24-
command: test
25-
args: --verbose --features strict
24+
- name: Run Test Script
25+
env: ${{ matrix.rust }}
26+
run: ./contrib/test.sh
2627

2728
fmt:
2829
name: Rustfmt
@@ -60,7 +61,8 @@ jobs:
6061
command: clippy
6162
args: -- -D warnings
6263

63-
Embedded:
64+
EmbeddedWithAlloc:
65+
name: no_std with alloc
6466
runs-on: ubuntu-latest
6567
steps:
6668
- name: Checkout
@@ -79,4 +81,20 @@ jobs:
7981
env:
8082
RUSTFLAGS: "-C link-arg=-Tlink.x"
8183
CARGO_TARGET_THUMBV7M_NONE_EABI_RUNNER: "qemu-system-arm -cpu cortex-m3 -machine mps2-an385 -nographic -semihosting-config enable=on,target=native -kernel"
82-
run: cd embedded && cargo run --target thumbv7m-none-eabi
84+
run: cd embedded/with-allocator && cargo run --target thumbv7m-none-eabi
85+
86+
EmbeddedNoAlloc:
87+
name: no_std no alloc
88+
runs-on: ubuntu-latest
89+
strategy:
90+
steps:
91+
- uses: actions/checkout@v2
92+
- uses: actions-rs/toolchain@v1
93+
with:
94+
profile: minimal
95+
toolchain: stable
96+
override: true
97+
- uses: actions-rs/cargo@v1
98+
with:
99+
command: rustc
100+
args: -- -C link-arg=-nostartfiles

.gitignore

+4-2
Original file line numberDiff line numberDiff line change
@@ -2,5 +2,7 @@
22
*.o
33
/Cargo.lock
44

5-
/embedded/Cargo.lock
6-
/embedded/.cargo
5+
/embedded/no-allocator/Cargo.lock
6+
/embedded/no-allocator/.cargo
7+
/embedded/with-allocator/Cargo.lock
8+
/embedded/with-allocator/.cargo

Cargo.toml

+3-1
Original file line numberDiff line numberDiff line change
@@ -13,7 +13,9 @@ edition = "2018"
1313

1414
[features]
1515
default = ["std"]
16-
std = []
16+
std = ["alloc"]
17+
alloc = []
18+
1719
# Only for CI to make all warnings errors, do not activate otherwise (may break forward compatibility)
1820
strict = []
1921

contrib/test.sh

+25
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,25 @@
1+
#!/bin/sh
2+
#
3+
# CI test script for rust-bech32.
4+
#
5+
# The "strict" feature is used to configure cargo to deny all warnings, always use it in test runs.
6+
7+
set -ex
8+
9+
# Sanity, check tools exist.
10+
cargo --version
11+
rustc --version
12+
13+
# Check without features ("strict" is a CI feature only, see above).
14+
cargo build --no-default-features --features="strict"
15+
cargo test --no-default-features --features="strict"
16+
17+
# Check "std" feature (implies "alloc").
18+
cargo build --no-default-features --features="strict std"
19+
cargo test --no-default-features --features="strict std"
20+
21+
# Check "alloc" feature alone.
22+
cargo build --no-default-features --features="strict alloc"
23+
cargo test --no-default-features --features="strict alloc"
24+
25+
exit 0

embedded/no-allocator/Cargo.toml

+15
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,15 @@
1+
[package]
2+
authors = ["Tobin C. Harding <[email protected]>"]
3+
edition = "2018"
4+
readme = "README.md"
5+
name = "no-allocator"
6+
version = "0.1.0"
7+
8+
[profile.dev]
9+
panic = "abort"
10+
11+
[profile.release]
12+
panic = "abort"
13+
14+
[dependencies]
15+
bech32 = { path = "../../", default_features = false }

embedded/no-allocator/README.md

+10
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,10 @@
1+
# no_std test crate without an allocator
2+
3+
This crate is based on the blog post found at:
4+
5+
https://blog.dbrgn.ch/2019/12/24/testing-for-no-std-compatibility/
6+
7+
Its purpose is to test that the `rust-bech32` library can be built in a `no_std` environment without
8+
a global allocator.
9+
10+
Build with: `cargo rustc -- -C link-arg=-nostartfiles`.

embedded/no-allocator/src/main.rs

+26
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,26 @@
1+
//! Test `no_std` build of `bech32`.
2+
//!
3+
//! Build with: `cargo rustc -- -C link-arg=-nostartfiles`.
4+
//!
5+
6+
#![no_std]
7+
#![no_main]
8+
9+
use core::panic::PanicInfo;
10+
11+
// Note: `#[global_allocator]` is NOT set.
12+
13+
#[allow(unused_imports)]
14+
use bech32;
15+
16+
/// This function is called on panic, defining this ensures build will fail if `std` is enabled
17+
/// because `panic` will be defined twice.
18+
#[panic_handler]
19+
fn panic(_info: &PanicInfo) -> ! {
20+
loop {}
21+
}
22+
23+
#[no_mangle]
24+
pub extern "C" fn _start() -> ! {
25+
loop {}
26+
}

embedded/Cargo.toml embedded/with-allocator/Cargo.toml

+3-3
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,7 @@
22
authors = ["Riccardo Casatta <[email protected]>"]
33
edition = "2018"
44
readme = "README.md"
5-
name = "embedded"
5+
name = "with-allocator"
66
version = "0.1.0"
77

88
[dependencies]
@@ -11,10 +11,10 @@ cortex-m-rt = "0.6.10"
1111
cortex-m-semihosting = "0.3.3"
1212
panic-halt = "0.2.0"
1313
alloc-cortex-m = "0.4.1"
14-
bech32 = { path="../", default-features = false }
14+
bech32 = { path="../../", default-features = false }
1515

1616
[[bin]]
17-
name = "embedded"
17+
name = "with-allocator"
1818
test = false
1919
bench = false
2020

File renamed without changes.
File renamed without changes.

src/lib.rs

+30-5
Original file line numberDiff line numberDiff line change
@@ -57,19 +57,21 @@ assert_eq!(variant, Variant::Bech32);
5757
#![cfg_attr(feature = "strict", deny(warnings))]
5858
#![cfg_attr(all(not(feature = "std"), not(test)), no_std)]
5959

60-
#[cfg(all(not(feature = "std"), not(test)))]
60+
#[cfg(feature = "alloc")]
6161
extern crate alloc;
6262

6363
#[cfg(any(test, feature = "std"))]
6464
extern crate core;
6565

66-
#[cfg(all(not(feature = "std"), not(test)))]
66+
#[cfg(all(feature = "alloc", not(feature = "std")))]
6767
use alloc::borrow::Cow;
68-
#[cfg(all(not(feature = "std"), not(test)))]
68+
#[cfg(all(feature = "alloc", not(feature = "std")))]
6969
use alloc::{string::String, vec::Vec};
70-
use core::convert::{Infallible, TryFrom};
70+
#[cfg(feature = "alloc")]
71+
use core::convert::Infallible;
72+
use core::convert::TryFrom;
7173
use core::{fmt, mem};
72-
#[cfg(any(feature = "std", test))]
74+
#[cfg(feature = "std")]
7375
use std::borrow::Cow;
7476

7577
/// Integer in the range `0..32`
@@ -220,6 +222,7 @@ pub trait FromBase32: Sized {
220222
fn from_base32(b32: &[u5]) -> Result<Self, Self::Err>;
221223
}
222224

225+
#[cfg(feature = "alloc")]
223226
impl WriteBase32 for Vec<u5> {
224227
type Err = Infallible;
225228

@@ -234,6 +237,7 @@ impl WriteBase32 for Vec<u5> {
234237
}
235238
}
236239

240+
#[cfg(feature = "alloc")]
237241
impl FromBase32 for Vec<u8> {
238242
type Err = Error;
239243

@@ -243,6 +247,7 @@ impl FromBase32 for Vec<u8> {
243247
}
244248

245249
/// A trait for converting a value to a type `T` that represents a `u5` slice.
250+
#[cfg(feature = "alloc")]
246251
pub trait ToBase32 {
247252
/// Convert `Self` to base32 vector
248253
fn to_base32(&self) -> Vec<u5> {
@@ -257,11 +262,13 @@ pub trait ToBase32 {
257262
}
258263

259264
/// Interface to calculate the length of the base32 representation before actually serializing
265+
#[cfg(feature = "alloc")]
260266
pub trait Base32Len: ToBase32 {
261267
/// Calculate the base32 serialized length
262268
fn base32_len(&self) -> usize;
263269
}
264270

271+
#[cfg(feature = "alloc")]
265272
impl<T: AsRef<[u8]>> ToBase32 for T {
266273
fn write_base32<W: WriteBase32>(&self, writer: &mut W) -> Result<(), <W as WriteBase32>::Err> {
267274
// Amount of bits left over from last round, stored in buffer.
@@ -306,6 +313,7 @@ impl<T: AsRef<[u8]>> ToBase32 for T {
306313
}
307314
}
308315

316+
#[cfg(feature = "alloc")]
309317
impl<T: AsRef<[u8]>> Base32Len for T {
310318
fn base32_len(&self) -> usize {
311319
let bits = self.as_ref().len() * 8;
@@ -327,6 +335,7 @@ pub trait CheckBase32<T: AsRef<[u5]>> {
327335
fn check_base32(self) -> Result<T, Self::Err>;
328336
}
329337

338+
#[cfg(feature = "alloc")]
330339
impl<T: AsRef<[u8]>> CheckBase32<Vec<u5>> for T {
331340
type Err = Error;
332341

@@ -336,6 +345,7 @@ impl<T: AsRef<[u8]>> CheckBase32<Vec<u5>> for T {
336345
}
337346

338347
#[derive(Clone, Copy, PartialEq, Eq)]
348+
#[cfg(feature = "alloc")]
339349
enum Case {
340350
Upper,
341351
Lower,
@@ -348,6 +358,7 @@ enum Case {
348358
/// * **MixedCase**: If the HRP contains both uppercase and lowercase characters.
349359
/// * **InvalidChar**: If the HRP contains any non-ASCII characters (outside 33..=126).
350360
/// * **InvalidLength**: If the HRP is outside 1..83 characters long.
361+
#[cfg(feature = "alloc")]
351362
fn check_hrp(hrp: &str) -> Result<Case, Error> {
352363
if hrp.is_empty() || hrp.len() > 83 {
353364
return Err(Error::InvalidLength);
@@ -387,6 +398,7 @@ fn check_hrp(hrp: &str) -> Result<Case, Error> {
387398
/// * If [check_hrp] returns an error for the given HRP.
388399
/// # Deviations from standard
389400
/// * No length limits are enforced for the data part
401+
#[cfg(feature = "alloc")]
390402
pub fn encode_to_fmt<T: AsRef<[u5]>>(
391403
fmt: &mut fmt::Write,
392404
hrp: &str,
@@ -416,6 +428,7 @@ pub fn encode_to_fmt<T: AsRef<[u5]>>(
416428
/// * If [check_hrp] returns an error for the given HRP.
417429
/// # Deviations from standard
418430
/// * No length limits are enforced for the data part
431+
#[cfg(feature = "alloc")]
419432
pub fn encode_without_checksum_to_fmt<T: AsRef<[u5]>>(
420433
fmt: &mut fmt::Write,
421434
hrp: &str,
@@ -454,6 +467,7 @@ const BECH32M_CONST: u32 = 0x2bc8_30a3;
454467

455468
impl Variant {
456469
// Produce the variant based on the remainder of the polymod operation
470+
#[cfg(feature = "alloc")]
457471
fn from_remainder(c: u32) -> Option<Self> {
458472
match c {
459473
BECH32_CONST => Some(Variant::Bech32),
@@ -476,6 +490,7 @@ impl Variant {
476490
/// * If [check_hrp] returns an error for the given HRP.
477491
/// # Deviations from standard
478492
/// * No length limits are enforced for the data part
493+
#[cfg(feature = "alloc")]
479494
pub fn encode<T: AsRef<[u5]>>(hrp: &str, data: T, variant: Variant) -> Result<String, Error> {
480495
let mut buf = String::new();
481496
encode_to_fmt(&mut buf, hrp, data, variant)?.unwrap();
@@ -488,6 +503,7 @@ pub fn encode<T: AsRef<[u5]>>(hrp: &str, data: T, variant: Variant) -> Result<St
488503
/// * If [check_hrp] returns an error for the given HRP.
489504
/// # Deviations from standard
490505
/// * No length limits are enforced for the data part
506+
#[cfg(feature = "alloc")]
491507
pub fn encode_without_checksum<T: AsRef<[u5]>>(hrp: &str, data: T) -> Result<String, Error> {
492508
let mut buf = String::new();
493509
encode_without_checksum_to_fmt(&mut buf, hrp, data)?.unwrap();
@@ -497,6 +513,7 @@ pub fn encode_without_checksum<T: AsRef<[u5]>>(hrp: &str, data: T) -> Result<Str
497513
/// Decode a bech32 string into the raw HRP and the data bytes.
498514
///
499515
/// Returns the HRP in lowercase, the data with the checksum removed, and the encoding.
516+
#[cfg(feature = "alloc")]
500517
pub fn decode(s: &str) -> Result<(String, Vec<u5>, Variant), Error> {
501518
let (hrp_lower, mut data) = split_and_decode(s)?;
502519
if data.len() < CHECKSUM_LENGTH {
@@ -518,9 +535,11 @@ pub fn decode(s: &str) -> Result<(String, Vec<u5>, Variant), Error> {
518535
/// Decode a bech32 string into the raw HRP and the data bytes, assuming no checksum.
519536
///
520537
/// Returns the HRP in lowercase and the data.
538+
#[cfg(feature = "alloc")]
521539
pub fn decode_without_checksum(s: &str) -> Result<(String, Vec<u5>), Error> { split_and_decode(s) }
522540

523541
/// Decode a bech32 string into the raw HRP and the `u5` data.
542+
#[cfg(feature = "alloc")]
524543
fn split_and_decode(s: &str) -> Result<(String, Vec<u5>), Error> {
525544
// Split at separator and check for two pieces
526545
let (raw_hrp, raw_data) = match s.rfind(SEP) {
@@ -577,12 +596,14 @@ fn split_and_decode(s: &str) -> Result<(String, Vec<u5>), Error> {
577596
Ok((hrp_lower, data))
578597
}
579598

599+
#[cfg(feature = "alloc")]
580600
fn verify_checksum(hrp: &[u8], data: &[u5]) -> Option<Variant> {
581601
let mut exp = hrp_expand(hrp);
582602
exp.extend_from_slice(data);
583603
Variant::from_remainder(polymod(&exp))
584604
}
585605

606+
#[cfg(feature = "alloc")]
586607
fn hrp_expand(hrp: &[u8]) -> Vec<u5> {
587608
let mut v: Vec<u5> = Vec::new();
588609
for b in hrp {
@@ -595,6 +616,7 @@ fn hrp_expand(hrp: &[u8]) -> Vec<u5> {
595616
v
596617
}
597618

619+
#[cfg(feature = "alloc")]
598620
fn polymod(values: &[u5]) -> u32 {
599621
let mut chk: u32 = 1;
600622
let mut b: u8;
@@ -623,6 +645,7 @@ const CHARSET: [char; 32] = [
623645
];
624646

625647
/// Reverse character set. Maps ASCII byte -> CHARSET index on [0,31]
648+
#[cfg(feature = "alloc")]
626649
const CHARSET_REV: [i8; 128] = [
627650
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
628651
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
@@ -697,6 +720,7 @@ impl std::error::Error for Error {
697720
/// let base5 = convert_bits(&[0xff], 8, 5, true);
698721
/// assert_eq!(base5.unwrap(), vec![0x1f, 0x1c]);
699722
/// ```
723+
#[cfg(feature = "alloc")]
700724
pub fn convert_bits<T>(data: &[T], from: u32, to: u32, pad: bool) -> Result<Vec<u8>, Error>
701725
where
702726
T: Into<u8> + Copy,
@@ -732,6 +756,7 @@ where
732756
}
733757

734758
#[cfg(test)]
759+
#[cfg(feature = "alloc")] // Note, all the unit tests currently require an allocator.
735760
mod tests {
736761
use super::*;
737762

0 commit comments

Comments
 (0)