| /* SPDX-License-Identifier: LGPL-2.1-or-later */ |
| /* |
| * |
| * BlueZ - Bluetooth protocol stack for Linux |
| * |
| * Copyright (C) 2012-2014 Intel Corporation. All rights reserved. |
| * Copyright 2023-2024 NXP |
| * |
| * |
| */ |
| |
| #include <stdint.h> |
| #include <stdlib.h> |
| #include <stdarg.h> |
| #include <stdbool.h> |
| #include <alloca.h> |
| #include <byteswap.h> |
| #include <string.h> |
| #include <sys/types.h> |
| #include <sys/uio.h> |
| |
| #define ARRAY_SIZE(arr) (sizeof(arr) / sizeof((arr)[0])) |
| #define BIT(n) (1 << (n)) |
| |
| #undef MIN |
| #define MIN(a, b) ((a) < (b) ? (a) : (b)) |
| |
| #undef MAX |
| #define MAX(a, b) ((a) > (b) ? (a) : (b)) |
| |
| #if __BYTE_ORDER == __LITTLE_ENDIAN |
| #define le16_to_cpu(val) (val) |
| #define le32_to_cpu(val) (val) |
| #define le64_to_cpu(val) (val) |
| #define cpu_to_le16(val) (val) |
| #define cpu_to_le32(val) (val) |
| #define cpu_to_le64(val) (val) |
| #define be16_to_cpu(val) bswap_16(val) |
| #define be32_to_cpu(val) bswap_32(val) |
| #define be64_to_cpu(val) bswap_64(val) |
| #define cpu_to_be16(val) bswap_16(val) |
| #define cpu_to_be32(val) bswap_32(val) |
| #define cpu_to_be64(val) bswap_64(val) |
| #elif __BYTE_ORDER == __BIG_ENDIAN |
| #define le16_to_cpu(val) bswap_16(val) |
| #define le32_to_cpu(val) bswap_32(val) |
| #define le64_to_cpu(val) bswap_64(val) |
| #define cpu_to_le16(val) bswap_16(val) |
| #define cpu_to_le32(val) bswap_32(val) |
| #define cpu_to_le64(val) bswap_64(val) |
| #define be16_to_cpu(val) (val) |
| #define be32_to_cpu(val) (val) |
| #define be64_to_cpu(val) (val) |
| #define cpu_to_be16(val) (val) |
| #define cpu_to_be32(val) (val) |
| #define cpu_to_be64(val) (val) |
| #else |
| #error "Unknown byte order" |
| #endif |
| |
| #define get_unaligned(ptr) \ |
| __extension__ ({ \ |
| struct __attribute__((packed)) { \ |
| __typeof__(*(ptr)) __v; \ |
| } *__p = (__typeof__(__p)) (ptr); \ |
| __p->__v; \ |
| }) |
| |
| #define put_unaligned(val, ptr) \ |
| do { \ |
| struct __attribute__((packed)) { \ |
| __typeof__(*(ptr)) __v; \ |
| } *__p = (__typeof__(__p)) (ptr); \ |
| __p->__v = (val); \ |
| } while (0) |
| |
| #define PTR_TO_UINT(p) ((unsigned int) ((uintptr_t) (p))) |
| #define UINT_TO_PTR(u) ((void *) ((uintptr_t) (u))) |
| |
| #define PTR_TO_INT(p) ((int) ((intptr_t) (p))) |
| #define INT_TO_PTR(u) ((void *) ((intptr_t) (u))) |
| |
| #define new0(type, count) \ |
| (type *) (__extension__ ({ \ |
| size_t __n = (size_t) (count); \ |
| size_t __s = sizeof(type); \ |
| void *__p; \ |
| __p = util_malloc(__n * __s); \ |
| memset(__p, 0, __n * __s); \ |
| __p; \ |
| })) |
| |
| #define newa(t, n) ((t*) alloca(sizeof(t)*(n))) |
| #define malloc0(n) (calloc(1, (n))) |
| |
| char *strdelimit(char *str, char *del, char c); |
| int strsuffix(const char *str, const char *suffix); |
| char *strstrip(char *str); |
| |
| size_t strnlenutf8(const char *str, size_t len); |
| bool strisutf8(const char *str, size_t length); |
| bool argsisutf8(int argc, char *argv[]); |
| char *strtoutf8(char *str, size_t len); |
| |
| void *util_malloc(size_t size); |
| void *util_memdup(const void *src, size_t size); |
| |
| typedef void (*util_debug_func_t)(const char *str, void *user_data); |
| |
| void util_debug_va(util_debug_func_t function, void *user_data, |
| const char *format, va_list va); |
| |
| void util_debug(util_debug_func_t function, void *user_data, |
| const char *format, ...) |
| __attribute__((format(printf, 3, 4))); |
| |
| void util_hexdump(const char dir, const unsigned char *buf, size_t len, |
| util_debug_func_t function, void *user_data); |
| |
| #define UTIL_BIT_DEBUG(_bit, _str) \ |
| { \ |
| .bit = _bit, \ |
| .str = _str, \ |
| } |
| |
| struct util_bit_debugger { |
| uint64_t bit; |
| const char *str; |
| }; |
| |
| uint64_t util_debug_bit(const char *label, uint64_t val, |
| const struct util_bit_debugger *table, |
| util_debug_func_t func, void *user_data); |
| |
| #define UTIL_LTV_DEBUG(_type, _func) \ |
| { \ |
| .type = _type, \ |
| .func = _func, \ |
| } |
| |
| struct util_ltv_debugger { |
| uint8_t type; |
| void (*func)(const uint8_t *data, uint8_t len, |
| util_debug_func_t func, void *user_data); |
| }; |
| |
| void util_ltv_push(struct iovec *output, uint8_t l, uint8_t t, void *v); |
| |
| bool util_debug_ltv(const uint8_t *data, uint8_t len, |
| const struct util_ltv_debugger *debugger, size_t num, |
| util_debug_func_t function, void *user_data); |
| |
| typedef void (*util_ltv_func_t)(size_t i, uint8_t l, uint8_t t, uint8_t *v, |
| void *user_data); |
| |
| bool util_ltv_foreach(const uint8_t *data, uint8_t len, uint8_t *type, |
| util_ltv_func_t func, void *user_data); |
| |
| unsigned char util_get_dt(const char *parent, const char *name); |
| |
| ssize_t util_getrandom(void *buf, size_t buflen, unsigned int flags); |
| |
| uint8_t util_get_uid(uint64_t *bitmap, uint8_t max); |
| void util_clear_uid(uint64_t *bitmap, uint8_t id); |
| |
| #define util_data(args...) ((const unsigned char[]) { args }) |
| |
| #define UTIL_IOV_INIT(args...) \ |
| { \ |
| .iov_base = (void *)util_data(args), \ |
| .iov_len = sizeof(util_data(args)), \ |
| } |
| |
| struct iovec *util_iov_dup(const struct iovec *iov, size_t cnt); |
| int util_iov_memcmp(const struct iovec *iov1, const struct iovec *iov2); |
| void util_iov_memcpy(struct iovec *iov, void *src, size_t len); |
| void *util_iov_push(struct iovec *iov, size_t len); |
| void *util_iov_push_mem(struct iovec *iov, size_t len, const void *data); |
| void *util_iov_push_le64(struct iovec *iov, uint64_t val); |
| void *util_iov_push_be64(struct iovec *iov, uint64_t val); |
| void *util_iov_push_le32(struct iovec *iov, uint32_t val); |
| void *util_iov_push_be32(struct iovec *iov, uint32_t val); |
| void *util_iov_push_le24(struct iovec *iov, uint32_t val); |
| void *util_iov_push_be24(struct iovec *iov, uint32_t val); |
| void *util_iov_push_le16(struct iovec *iov, uint16_t val); |
| void *util_iov_push_be16(struct iovec *iov, uint16_t val); |
| void *util_iov_push_u8(struct iovec *iov, uint8_t val); |
| void *util_iov_append(struct iovec *iov, const void *data, size_t len); |
| struct iovec *util_iov_new(void *data, size_t len); |
| void *util_iov_pull(struct iovec *iov, size_t len); |
| void *util_iov_pull_mem(struct iovec *iov, size_t len); |
| void *util_iov_pull_le64(struct iovec *iov, uint64_t *val); |
| void *util_iov_pull_be64(struct iovec *iov, uint64_t *val); |
| void *util_iov_pull_le32(struct iovec *iov, uint32_t *val); |
| void *util_iov_pull_be32(struct iovec *iov, uint32_t *val); |
| void *util_iov_pull_le24(struct iovec *iov, uint32_t *val); |
| void *util_iov_pull_be24(struct iovec *iov, uint32_t *val); |
| void *util_iov_pull_le16(struct iovec *iov, uint16_t *val); |
| void *util_iov_pull_be16(struct iovec *iov, uint16_t *val); |
| void *util_iov_pull_u8(struct iovec *iov, uint8_t *val); |
| void util_iov_free(struct iovec *iov, size_t cnt); |
| |
| const char *bt_uuid16_to_str(uint16_t uuid); |
| const char *bt_uuid32_to_str(uint32_t uuid); |
| const char *bt_uuid128_to_str(const uint8_t uuid[16]); |
| const char *bt_uuidstr_to_str(const char *uuid); |
| const char *bt_appear_to_str(uint16_t appearance); |
| |
| static inline int8_t get_s8(const void *ptr) |
| { |
| return *((int8_t *) ptr); |
| } |
| |
| static inline uint8_t get_u8(const void *ptr) |
| { |
| return *((uint8_t *) ptr); |
| } |
| |
| static inline uint16_t get_le16(const void *ptr) |
| { |
| return le16_to_cpu(get_unaligned((const uint16_t *) ptr)); |
| } |
| |
| static inline uint16_t get_be16(const void *ptr) |
| { |
| return be16_to_cpu(get_unaligned((const uint16_t *) ptr)); |
| } |
| |
| static inline uint32_t get_le24(const void *ptr) |
| { |
| const uint8_t *src = ptr; |
| |
| return ((uint32_t)src[2] << 16) | get_le16(ptr); |
| } |
| |
| static inline uint32_t get_be24(const void *ptr) |
| { |
| const uint8_t *src = ptr; |
| |
| return ((uint32_t)src[0] << 16) | get_be16(&src[1]); |
| } |
| |
| static inline uint32_t get_le32(const void *ptr) |
| { |
| return le32_to_cpu(get_unaligned((const uint32_t *) ptr)); |
| } |
| |
| static inline uint32_t get_be32(const void *ptr) |
| { |
| return be32_to_cpu(get_unaligned((const uint32_t *) ptr)); |
| } |
| |
| static inline uint64_t get_le64(const void *ptr) |
| { |
| return le64_to_cpu(get_unaligned((const uint64_t *) ptr)); |
| } |
| |
| static inline uint64_t get_be64(const void *ptr) |
| { |
| return be64_to_cpu(get_unaligned((const uint64_t *) ptr)); |
| } |
| |
| static inline void put_u8(uint8_t val, void *dst) |
| { |
| put_unaligned(val, (uint8_t *) dst); |
| } |
| |
| static inline void put_le16(uint16_t val, void *dst) |
| { |
| put_unaligned(cpu_to_le16(val), (uint16_t *) dst); |
| } |
| |
| static inline void put_be16(uint16_t val, const void *ptr) |
| { |
| put_unaligned(cpu_to_be16(val), (uint16_t *) ptr); |
| } |
| |
| static inline void put_le24(uint32_t val, void *ptr) |
| { |
| put_le16(val, ptr); |
| put_unaligned(val >> 16, (uint8_t *) ptr + 2); |
| } |
| |
| static inline void put_be24(uint32_t val, void *ptr) |
| { |
| put_unaligned(val >> 16, (uint8_t *) ptr + 2); |
| put_be16(val, ptr + 1); |
| } |
| |
| static inline void put_le32(uint32_t val, void *dst) |
| { |
| put_unaligned(cpu_to_le32(val), (uint32_t *) dst); |
| } |
| |
| static inline void put_be32(uint32_t val, void *dst) |
| { |
| put_unaligned(cpu_to_be32(val), (uint32_t *) dst); |
| } |
| |
| static inline void put_le64(uint64_t val, void *dst) |
| { |
| put_unaligned(cpu_to_le64(val), (uint64_t *) dst); |
| } |
| |
| static inline void put_be64(uint64_t val, void *dst) |
| { |
| put_unaligned(cpu_to_be64(val), (uint64_t *) dst); |
| } |