Commit e7d4a95d authored by Johannes Berg's avatar Johannes Berg Committed by Kalle Valo

bitfield: fix *_encode_bits()

There's a bug in *_encode_bits() in using ~field_multiplier() for
the check whether or not the constant value fits into the field,
this is wrong and clearly ~field_mask() was intended. This was
triggering for me for both constant and non-constant values.

Additionally, make this case actually into an compile error.
Declaring the extern function that will never exist with just a
warning is pointless as then later we'll just get a link error.

While at it, also fix the indentation in those lines I'm touching.

Finally, as suggested by Andy Shevchenko, add some tests and for
that introduce also u8 helpers. The tests don't compile without
the fix, showing that it's necessary.

Fixes: 00b0c9b8 ("Add primitives for manipulating bitfields both in host- and fixed-endian.")
Reviewed-by: default avatarAndy Shevchenko <andy.shevchenko@gmail.com>
Signed-off-by: default avatarJohannes Berg <johannes@sipsolutions.net>
Signed-off-by: default avatarKalle Valo <kvalo@codeaurora.org>
parent ab8d9046
...@@ -104,7 +104,7 @@ ...@@ -104,7 +104,7 @@
(typeof(_mask))(((_reg) & (_mask)) >> __bf_shf(_mask)); \ (typeof(_mask))(((_reg) & (_mask)) >> __bf_shf(_mask)); \
}) })
extern void __compiletime_warning("value doesn't fit into mask") extern void __compiletime_error("value doesn't fit into mask")
__field_overflow(void); __field_overflow(void);
extern void __compiletime_error("bad bitfield mask") extern void __compiletime_error("bad bitfield mask")
__bad_mask(void); __bad_mask(void);
...@@ -121,8 +121,8 @@ static __always_inline u64 field_mask(u64 field) ...@@ -121,8 +121,8 @@ static __always_inline u64 field_mask(u64 field)
#define ____MAKE_OP(type,base,to,from) \ #define ____MAKE_OP(type,base,to,from) \
static __always_inline __##type type##_encode_bits(base v, base field) \ static __always_inline __##type type##_encode_bits(base v, base field) \
{ \ { \
if (__builtin_constant_p(v) && (v & ~field_multiplier(field))) \ if (__builtin_constant_p(v) && (v & ~field_mask(field))) \
__field_overflow(); \ __field_overflow(); \
return to((v & field_mask(field)) * field_multiplier(field)); \ return to((v & field_mask(field)) * field_multiplier(field)); \
} \ } \
static __always_inline __##type type##_replace_bits(__##type old, \ static __always_inline __##type type##_replace_bits(__##type old, \
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment