Skip to content

Commit a8b1a09

Browse files
Add support for enums with negative discriminants (#4204)
1 parent 5f70a71 commit a8b1a09

File tree

15 files changed

+207
-62
lines changed

15 files changed

+207
-62
lines changed

CHANGELOG.md

+3
Original file line numberDiff line numberDiff line change
@@ -14,6 +14,9 @@
1414
* Added JSDoc type annotations to C-style enums.
1515
[#4192](https://github.com/rustwasm/wasm-bindgen/pull/4192)
1616

17+
* Added support for C-style enums with negative discriminants.
18+
[#4204](https://github.com/rustwasm/wasm-bindgen/pull/4204)
19+
1720
### Changed
1821

1922
* String enums now generate private TypeScript types but only if used.

crates/backend/src/ast.rs

+3
Original file line numberDiff line numberDiff line change
@@ -445,6 +445,9 @@ pub struct Enum {
445445
pub rust_name: Ident,
446446
/// The name of this enum in JS code
447447
pub js_name: String,
448+
/// Whether the variant values and hole are signed, meaning that they
449+
/// represent the bits of a `i32` value.
450+
pub signed: bool,
448451
/// The variants provided by this enum
449452
pub variants: Vec<Variant>,
450453
/// The doc comments on this enum, if any

crates/backend/src/codegen.rs

+15-10
Original file line numberDiff line numberDiff line change
@@ -1535,10 +1535,15 @@ impl ToTokens for ast::Enum {
15351535
let name_len = name_str.len() as u32;
15361536
let name_chars = name_str.chars().map(|c| c as u32);
15371537
let hole = &self.hole;
1538+
let underlying = if self.signed {
1539+
quote! { i32 }
1540+
} else {
1541+
quote! { u32 }
1542+
};
15381543
let cast_clauses = self.variants.iter().map(|variant| {
15391544
let variant_name = &variant.name;
15401545
quote! {
1541-
if js == #enum_name::#variant_name as u32 {
1546+
if js == #enum_name::#variant_name as #underlying {
15421547
#enum_name::#variant_name
15431548
}
15441549
}
@@ -1548,20 +1553,20 @@ impl ToTokens for ast::Enum {
15481553
(quote! {
15491554
#[automatically_derived]
15501555
impl #wasm_bindgen::convert::IntoWasmAbi for #enum_name {
1551-
type Abi = u32;
1556+
type Abi = #underlying;
15521557

15531558
#[inline]
1554-
fn into_abi(self) -> u32 {
1555-
self as u32
1559+
fn into_abi(self) -> #underlying {
1560+
self as #underlying
15561561
}
15571562
}
15581563

15591564
#[automatically_derived]
15601565
impl #wasm_bindgen::convert::FromWasmAbi for #enum_name {
1561-
type Abi = u32;
1566+
type Abi = #underlying;
15621567

15631568
#[inline]
1564-
unsafe fn from_abi(js: u32) -> Self {
1569+
unsafe fn from_abi(js: #underlying) -> Self {
15651570
#(#cast_clauses else)* {
15661571
#wasm_bindgen::throw_str("invalid enum value passed")
15671572
}
@@ -1571,13 +1576,13 @@ impl ToTokens for ast::Enum {
15711576
#[automatically_derived]
15721577
impl #wasm_bindgen::convert::OptionFromWasmAbi for #enum_name {
15731578
#[inline]
1574-
fn is_none(val: &u32) -> bool { *val == #hole }
1579+
fn is_none(val: &Self::Abi) -> bool { *val == #hole as #underlying }
15751580
}
15761581

15771582
#[automatically_derived]
15781583
impl #wasm_bindgen::convert::OptionIntoWasmAbi for #enum_name {
15791584
#[inline]
1580-
fn none() -> Self::Abi { #hole }
1585+
fn none() -> Self::Abi { #hole as #underlying }
15811586
}
15821587

15831588
#[automatically_derived]
@@ -1597,7 +1602,7 @@ impl ToTokens for ast::Enum {
15971602
#wasm_bindgen::JsValue
15981603
{
15991604
fn from(value: #enum_name) -> Self {
1600-
#wasm_bindgen::JsValue::from_f64((value as u32).into())
1605+
#wasm_bindgen::JsValue::from_f64((value as #underlying).into())
16011606
}
16021607
}
16031608

@@ -1608,7 +1613,7 @@ impl ToTokens for ast::Enum {
16081613
fn try_from_js_value(value: #wasm_bindgen::JsValue)
16091614
-> #wasm_bindgen::__rt::core::result::Result<Self, <#enum_name as #wasm_bindgen::convert::TryFromJsValue>::Error> {
16101615
use #wasm_bindgen::__rt::core::convert::TryFrom;
1611-
let js = f64::try_from(&value)? as u32;
1616+
let js = f64::try_from(&value)? as #underlying;
16121617

16131618
#wasm_bindgen::__rt::core::result::Result::Ok(
16141619
#(#try_from_cast_clauses else)* {

crates/backend/src/encode.rs

+1
Original file line numberDiff line numberDiff line change
@@ -239,6 +239,7 @@ fn shared_function<'a>(func: &'a ast::Function, _intern: &'a Interner) -> Functi
239239
fn shared_enum<'a>(e: &'a ast::Enum, intern: &'a Interner) -> Enum<'a> {
240240
Enum {
241241
name: &e.js_name,
242+
signed: e.signed,
242243
variants: e
243244
.variants
244245
.iter()

crates/cli-support/src/wit/mod.rs

+7-5
Original file line numberDiff line numberDiff line change
@@ -894,18 +894,20 @@ impl<'a> Context<'a> {
894894
}
895895

896896
fn enum_(&mut self, enum_: decode::Enum<'_>) -> Result<(), Error> {
897+
let signed = enum_.signed;
897898
let aux = AuxEnum {
898899
name: enum_.name.to_string(),
899900
comments: concatenate_comments(&enum_.comments),
900901
variants: enum_
901902
.variants
902903
.iter()
903904
.map(|v| {
904-
(
905-
v.name.to_string(),
906-
v.value,
907-
concatenate_comments(&v.comments),
908-
)
905+
let value = if signed {
906+
v.value as i32 as i64
907+
} else {
908+
v.value as i64
909+
};
910+
(v.name.to_string(), value, concatenate_comments(&v.comments))
909911
})
910912
.collect(),
911913
generate_typescript: enum_.generate_typescript,

crates/cli-support/src/wit/nonstandard.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -170,7 +170,7 @@ pub struct AuxEnum {
170170
pub comments: String,
171171
/// A list of variants with their name, value and comments
172172
/// and whether typescript bindings should be generated for each variant
173-
pub variants: Vec<(String, u32, String)>,
173+
pub variants: Vec<(String, i64, String)>,
174174
/// Whether typescript bindings should be generated for this enum.
175175
pub generate_typescript: bool,
176176
}

crates/cli/tests/reference/enums.d.ts

+9
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,7 @@ export function enum_echo(color: Color): Color;
44
export function option_enum_echo(color?: Color): Color | undefined;
55
export function get_name(color: Color): ColorName;
66
export function option_string_enum_echo(color?: ColorName): ColorName | undefined;
7+
export function option_order(order?: Ordering): Ordering | undefined;
78
/**
89
* A color.
910
*/
@@ -27,6 +28,14 @@ export enum ImplicitDiscriminant {
2728
C = 42,
2829
D = 43,
2930
}
31+
/**
32+
* A C-style enum with negative discriminants.
33+
*/
34+
export enum Ordering {
35+
Less = -1,
36+
Equal = 0,
37+
Greater = 1,
38+
}
3039
/**
3140
* The name of a color.
3241
*/

crates/cli/tests/reference/enums.js

+18
Original file line numberDiff line numberDiff line change
@@ -62,6 +62,15 @@ export function option_string_enum_echo(color) {
6262
return __wbindgen_enum_ColorName[ret];
6363
}
6464

65+
/**
66+
* @param {Ordering | undefined} [order]
67+
* @returns {Ordering | undefined}
68+
*/
69+
export function option_order(order) {
70+
const ret = wasm.option_order(isLikeNone(order) ? 2 : order);
71+
return ret === 2 ? undefined : ret;
72+
}
73+
6574
/**
6675
* A color.
6776
* @enum {0 | 1 | 2}
@@ -89,6 +98,15 @@ export const ImplicitDiscriminant = Object.freeze({
8998
C: 42, "42": "C",
9099
D: 43, "43": "D",
91100
});
101+
/**
102+
* A C-style enum with negative discriminants.
103+
* @enum {-1 | 0 | 1}
104+
*/
105+
export const Ordering = Object.freeze({
106+
Less: -1, "-1": "Less",
107+
Equal: 0, "0": "Equal",
108+
Greater: 1, "1": "Greater",
109+
});
92110

93111
const __wbindgen_enum_ColorName = ["green", "yellow", "red"];
94112

crates/cli/tests/reference/enums.rs

+13
Original file line numberDiff line numberDiff line change
@@ -65,3 +65,16 @@ pub enum ImplicitDiscriminant {
6565
C = 42,
6666
D,
6767
}
68+
69+
/// A C-style enum with negative discriminants.
70+
#[wasm_bindgen]
71+
pub enum Ordering {
72+
Less = -1,
73+
Equal = 0,
74+
Greater = 1,
75+
}
76+
77+
#[wasm_bindgen]
78+
pub fn option_order(order: Option<Ordering>) -> Option<Ordering> {
79+
order
80+
}

crates/cli/tests/reference/enums.wat

+2
Original file line numberDiff line numberDiff line change
@@ -4,12 +4,14 @@
44
(func $option_enum_echo (;1;) (type 0) (param i32) (result i32))
55
(func $get_name (;2;) (type 0) (param i32) (result i32))
66
(func $option_string_enum_echo (;3;) (type 0) (param i32) (result i32))
7+
(func $option_order (;4;) (type 0) (param i32) (result i32))
78
(memory (;0;) 17)
89
(export "memory" (memory 0))
910
(export "enum_echo" (func $enum_echo))
1011
(export "option_enum_echo" (func $option_enum_echo))
1112
(export "get_name" (func $get_name))
1213
(export "option_string_enum_echo" (func $option_string_enum_echo))
14+
(export "option_order" (func $option_order))
1315
(@custom "target_features" (after code) "\04+\0amultivalue+\0fmutable-globals+\0freference-types+\08sign-ext")
1416
)
1517

0 commit comments

Comments
 (0)