ffs.h 1.8 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889
  1. /*
  2. * Copyright (c) 2015, Wind River Systems, Inc.
  3. * Copyright (c) 2017, Oticon A/S
  4. *
  5. * SPDX-License-Identifier: Apache-2.0
  6. */
  7. #ifndef ZEPHYR_INCLUDE_ARCH_COMMON_FFS_H_
  8. #define ZEPHYR_INCLUDE_ARCH_COMMON_FFS_H_
  9. #ifndef _ASMLANGUAGE
  10. #include <zephyr/types.h>
  11. #ifdef __cplusplus
  12. extern "C" {
  13. #endif
  14. /**
  15. *
  16. * @brief find most significant bit set in a 32-bit word
  17. *
  18. * This routine finds the first bit set starting from the most significant bit
  19. * in the argument passed in and returns the index of that bit. Bits are
  20. * numbered starting at 1 from the least significant bit. A return value of
  21. * zero indicates that the value passed is zero.
  22. *
  23. * @return most significant bit set, 0 if @a op is 0
  24. */
  25. static ALWAYS_INLINE unsigned int find_msb_set(uint32_t op)
  26. {
  27. if (op == 0) {
  28. return 0;
  29. }
  30. return 32 - __builtin_clz(op);
  31. }
  32. /**
  33. *
  34. * @brief find least significant bit set in a 32-bit word
  35. *
  36. * This routine finds the first bit set starting from the least significant bit
  37. * in the argument passed in and returns the index of that bit. Bits are
  38. * numbered starting at 1 from the least significant bit. A return value of
  39. * zero indicates that the value passed is zero.
  40. *
  41. * @return least significant bit set, 0 if @a op is 0
  42. */
  43. static ALWAYS_INLINE unsigned int find_lsb_set(uint32_t op)
  44. {
  45. #ifdef CONFIG_TOOLCHAIN_HAS_BUILTIN_FFS
  46. return __builtin_ffs(op);
  47. #else
  48. /*
  49. * Toolchain does not have __builtin_ffs().
  50. * Need to do this manually.
  51. */
  52. int bit;
  53. if (op == 0) {
  54. return 0;
  55. }
  56. for (bit = 0; bit < 32; bit++) {
  57. if ((op & (1 << bit)) != 0) {
  58. return (bit + 1);
  59. }
  60. }
  61. /*
  62. * This should never happen but we need to keep
  63. * compiler happy.
  64. */
  65. return 0;
  66. #endif /* CONFIG_TOOLCHAIN_HAS_BUILTIN_FFS */
  67. }
  68. #ifdef __cplusplus
  69. }
  70. #endif
  71. #endif /* _ASMLANGUAGE */
  72. #endif /* ZEPHYR_INCLUDE_ARCH_COMMON_FFS_H_ */