276 lines
		
	
	
		
			9.4 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			276 lines
		
	
	
		
			9.4 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /* 
 | |
|  * The MIT License (MIT)
 | |
|  *
 | |
|  * Copyright (c) 2019 Ha Thach (tinyusb.org)
 | |
|  *
 | |
|  * Permission is hereby granted, free of charge, to any person obtaining a copy
 | |
|  * of this software and associated documentation files (the "Software"), to deal
 | |
|  * in the Software without restriction, including without limitation the rights
 | |
|  * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
 | |
|  * copies of the Software, and to permit persons to whom the Software is
 | |
|  * furnished to do so, subject to the following conditions:
 | |
|  *
 | |
|  * The above copyright notice and this permission notice shall be included in
 | |
|  * all copies or substantial portions of the Software.
 | |
|  *
 | |
|  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 | |
|  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 | |
|  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
 | |
|  * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
 | |
|  * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
 | |
|  * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
 | |
|  * THE SOFTWARE.
 | |
|  *
 | |
|  * This file is part of the TinyUSB stack.
 | |
|  */
 | |
| 
 | |
| /** \ingroup Group_Common
 | |
|  *  \defgroup Group_CommonH common.h
 | |
|  *  @{ */
 | |
| 
 | |
| #ifndef _TUSB_COMMON_H_
 | |
| #define _TUSB_COMMON_H_
 | |
| 
 | |
| #ifdef __cplusplus
 | |
|  extern "C" {
 | |
| #endif
 | |
| 
 | |
| //--------------------------------------------------------------------+
 | |
| // Macros Helper
 | |
| //--------------------------------------------------------------------+
 | |
| #define TU_ARRAY_SIZE(_arr)   ( sizeof(_arr) / sizeof(_arr[0]) )
 | |
| #define TU_MIN(_x, _y)        ( ( (_x) < (_y) ) ? (_x) : (_y) )
 | |
| #define TU_MAX(_x, _y)        ( ( (_x) > (_y) ) ? (_x) : (_y) )
 | |
| 
 | |
| #define TU_U16_HIGH(u16)      ((uint8_t) (((u16) >> 8) & 0x00ff))
 | |
| #define TU_U16_LOW(u16)       ((uint8_t) ((u16)       & 0x00ff))
 | |
| #define U16_TO_U8S_BE(u16)    TU_U16_HIGH(u16), TU_U16_LOW(u16)
 | |
| #define U16_TO_U8S_LE(u16)    TU_U16_LOW(u16), TU_U16_HIGH(u16)
 | |
| 
 | |
| #define U32_B1_U8(u32)        ((uint8_t) (((u32) >> 24) & 0x000000ff)) // MSB
 | |
| #define U32_B2_U8(u32)        ((uint8_t) (((u32) >> 16) & 0x000000ff))
 | |
| #define U32_B3_U8(u32)        ((uint8_t) (((u32) >>  8) & 0x000000ff))
 | |
| #define U32_B4_U8(u32)        ((uint8_t) ((u32)         & 0x000000ff)) // LSB
 | |
| 
 | |
| #define U32_TO_U8S_BE(u32)    U32_B1_U8(u32), U32_B2_U8(u32), U32_B3_U8(u32), U32_B4_U8(u32)
 | |
| #define U32_TO_U8S_LE(u32)    U32_B4_U8(u32), U32_B3_U8(u32), U32_B2_U8(u32), U32_B1_U8(u32)
 | |
| 
 | |
| #define TU_BIT(n)             (1U << (n))
 | |
| 
 | |
| //--------------------------------------------------------------------+
 | |
| // Includes
 | |
| //--------------------------------------------------------------------+
 | |
| 
 | |
| // Standard Headers
 | |
| #include <stdbool.h>
 | |
| #include <stdint.h>
 | |
| #include <stddef.h>
 | |
| #include <string.h>
 | |
| #include <stdio.h>
 | |
| 
 | |
| // Tinyusb Common Headers
 | |
| #include "tusb_option.h"
 | |
| #include "tusb_compiler.h"
 | |
| #include "tusb_verify.h"
 | |
| #include "tusb_error.h" // TODO remove
 | |
| #include "tusb_timeout.h"
 | |
| #include "tusb_types.h"
 | |
| 
 | |
| //--------------------------------------------------------------------+
 | |
| // Inline Functions
 | |
| //--------------------------------------------------------------------+
 | |
| #define tu_memclr(buffer, size)  memset((buffer), 0, (size))
 | |
| #define tu_varclr(_var)          tu_memclr(_var, sizeof(*(_var)))
 | |
| 
 | |
| static inline uint32_t tu_u32(uint8_t b1, uint8_t b2, uint8_t b3, uint8_t b4)
 | |
| {
 | |
|   return ( ((uint32_t) b1) << 24) + ( ((uint32_t) b2) << 16) + ( ((uint32_t) b3) << 8) + b4;
 | |
| }
 | |
| 
 | |
| static inline uint16_t tu_u16(uint8_t high, uint8_t low)
 | |
| {
 | |
|   return (uint16_t)((((uint16_t) high) << 8) + low);
 | |
| }
 | |
| 
 | |
| static inline uint8_t tu_u16_high(uint16_t u16) { return (uint8_t) (((uint16_t) (u16 >> 8)) & 0x00ff); }
 | |
| static inline uint8_t tu_u16_low (uint16_t u16) { return (uint8_t) (u16 & 0x00ff); }
 | |
| 
 | |
| // Min
 | |
| static inline uint8_t  tu_min8  (uint8_t  x, uint8_t y ) { return (x < y) ? x : y; }
 | |
| static inline uint16_t tu_min16 (uint16_t x, uint16_t y) { return (x < y) ? x : y; }
 | |
| static inline uint32_t tu_min32 (uint32_t x, uint32_t y) { return (x < y) ? x : y; }
 | |
| 
 | |
| // Max
 | |
| static inline uint8_t  tu_max8  (uint8_t  x, uint8_t y ) { return (x > y) ? x : y; }
 | |
| static inline uint16_t tu_max16 (uint16_t x, uint16_t y) { return (x > y) ? x : y; }
 | |
| static inline uint32_t tu_max32 (uint32_t x, uint32_t y) { return (x > y) ? x : y; }
 | |
| 
 | |
| // Align
 | |
| static inline uint32_t tu_align_n(uint32_t value, uint32_t alignment)
 | |
| {
 | |
|   return value & ((uint32_t) ~(alignment-1));
 | |
| }
 | |
| 
 | |
| static inline uint32_t tu_align32 (uint32_t value) { return (value & 0xFFFFFFE0UL); }
 | |
| static inline uint32_t tu_align16 (uint32_t value) { return (value & 0xFFFFFFF0UL); }
 | |
| static inline uint32_t tu_align4k (uint32_t value) { return (value & 0xFFFFF000UL); }
 | |
| static inline uint32_t tu_offset4k(uint32_t value) { return (value & 0xFFFUL); }
 | |
| 
 | |
| //------------- Mathematics -------------//
 | |
| static inline uint32_t tu_abs(int32_t value) { return (uint32_t)((value < 0) ? (-value) : value); }
 | |
| 
 | |
| /// inclusive range checking
 | |
| static inline bool tu_within(uint32_t lower, uint32_t value, uint32_t upper)
 | |
| {
 | |
|   return (lower <= value) && (value <= upper);
 | |
| }
 | |
| 
 | |
| // log2 of a value is its MSB's position
 | |
| // TODO use clz TODO remove
 | |
| static inline uint8_t tu_log2(uint32_t value)
 | |
| {
 | |
|   uint8_t result = 0;
 | |
| 
 | |
|   while (value >>= 1)
 | |
|   {
 | |
|     result++;
 | |
|   }
 | |
|   return result;
 | |
| }
 | |
| 
 | |
| // Bit
 | |
| static inline uint32_t tu_bit_set  (uint32_t value, uint8_t pos) { return value | TU_BIT(pos);                  }
 | |
| static inline uint32_t tu_bit_clear(uint32_t value, uint8_t pos) { return value & (~TU_BIT(pos));               }
 | |
| static inline bool     tu_bit_test (uint32_t value, uint8_t pos) { return (value & TU_BIT(pos)) ? true : false; }
 | |
| 
 | |
| /*------------------------------------------------------------------*/
 | |
| /* Count number of arguments of __VA_ARGS__
 | |
|  * - reference https://groups.google.com/forum/#!topic/comp.std.c/d-6Mj5Lko_s
 | |
|  * - _GET_NTH_ARG() takes args >= N (64) but only expand to Nth one (64th)
 | |
|  * - _RSEQ_N() is reverse sequential to N to add padding to have
 | |
|  * Nth position is the same as the number of arguments
 | |
|  * - ##__VA_ARGS__ is used to deal with 0 paramerter (swallows comma)
 | |
|  *------------------------------------------------------------------*/
 | |
| #ifndef TU_ARGS_NUM
 | |
| 
 | |
| #define TU_ARGS_NUM(...) 	 NARG_(_0, ##__VA_ARGS__,_RSEQ_N())
 | |
| 
 | |
| #define NARG_(...)        _GET_NTH_ARG(__VA_ARGS__)
 | |
| #define _GET_NTH_ARG( \
 | |
|           _1, _2, _3, _4, _5, _6, _7, _8, _9,_10, \
 | |
|          _11,_12,_13,_14,_15,_16,_17,_18,_19,_20, \
 | |
|          _21,_22,_23,_24,_25,_26,_27,_28,_29,_30, \
 | |
|          _31,_32,_33,_34,_35,_36,_37,_38,_39,_40, \
 | |
|          _41,_42,_43,_44,_45,_46,_47,_48,_49,_50, \
 | |
|          _51,_52,_53,_54,_55,_56,_57,_58,_59,_60, \
 | |
|          _61,_62,_63,N,...) N
 | |
| #define _RSEQ_N() \
 | |
|          62,61,60,                      \
 | |
|          59,58,57,56,55,54,53,52,51,50, \
 | |
|          49,48,47,46,45,44,43,42,41,40, \
 | |
|          39,38,37,36,35,34,33,32,31,30, \
 | |
|          29,28,27,26,25,24,23,22,21,20, \
 | |
|          19,18,17,16,15,14,13,12,11,10, \
 | |
|          9,8,7,6,5,4,3,2,1,0
 | |
| #endif
 | |
| 
 | |
| // To be removed
 | |
| //------------- Binary constant -------------//
 | |
| #if defined(__GNUC__) && !defined(__CC_ARM)
 | |
| 
 | |
| #define TU_BIN8(x)               ((uint8_t)  (0b##x))
 | |
| #define TU_BIN16(b1, b2)         ((uint16_t) (0b##b1##b2))
 | |
| #define TU_BIN32(b1, b2, b3, b4) ((uint32_t) (0b##b1##b2##b3##b4))
 | |
| 
 | |
| #else
 | |
| 
 | |
| //  internal macro of B8, B16, B32
 | |
| #define _B8__(x) (((x&0x0000000FUL)?1:0) \
 | |
|                 +((x&0x000000F0UL)?2:0) \
 | |
|                 +((x&0x00000F00UL)?4:0) \
 | |
|                 +((x&0x0000F000UL)?8:0) \
 | |
|                 +((x&0x000F0000UL)?16:0) \
 | |
|                 +((x&0x00F00000UL)?32:0) \
 | |
|                 +((x&0x0F000000UL)?64:0) \
 | |
|                 +((x&0xF0000000UL)?128:0))
 | |
| 
 | |
| #define TU_BIN8(d) ((uint8_t) _B8__(0x##d##UL))
 | |
| #define TU_BIN16(dmsb,dlsb) (((uint16_t)TU_BIN8(dmsb)<<8) + TU_BIN8(dlsb))
 | |
| #define TU_BIN32(dmsb,db2,db3,dlsb) \
 | |
|             (((uint32_t)TU_BIN8(dmsb)<<24) \
 | |
|             + ((uint32_t)TU_BIN8(db2)<<16) \
 | |
|             + ((uint32_t)TU_BIN8(db3)<<8) \
 | |
|             + TU_BIN8(dlsb))
 | |
| #endif
 | |
| 
 | |
| //--------------------------------------------------------------------+
 | |
| // Debug Function
 | |
| //--------------------------------------------------------------------+
 | |
| 
 | |
| // CFG_TUSB_DEBUG for debugging
 | |
| // 0 : no debug
 | |
| // 1 : print when there is error
 | |
| // 2 : print out log
 | |
| #if CFG_TUSB_DEBUG
 | |
| 
 | |
| void tu_print_mem(void const *buf, uint16_t count, uint8_t indent);
 | |
| 
 | |
| #ifndef tu_printf
 | |
|   #define tu_printf         printf
 | |
| #endif
 | |
| 
 | |
| // Log with debug level 1
 | |
| #define TU_LOG1               tu_printf
 | |
| #define TU_LOG1_MEM           tu_print_mem
 | |
| #define TU_LOG1_LOCATION()    tu_printf("%s: %d:\n", __PRETTY_FUNCTION__, __LINE__)
 | |
| 
 | |
| // Log with debug level 2
 | |
| #if CFG_TUSB_DEBUG > 1
 | |
|   #define TU_LOG2             TU_LOG1
 | |
|   #define TU_LOG2_MEM         TU_LOG1_MEM
 | |
|   #define TU_LOG2_LOCATION()  TU_LOG1_LOCATION()
 | |
| #endif
 | |
| 
 | |
| 
 | |
| typedef struct
 | |
| {
 | |
|   uint32_t key;
 | |
|   char const * data;
 | |
| }lookup_entry_t;
 | |
| 
 | |
| typedef struct
 | |
| {
 | |
|   uint16_t count;
 | |
|   lookup_entry_t const* items;
 | |
| } lookup_table_t;
 | |
| 
 | |
| static inline char const* lookup_find(lookup_table_t const* p_table, uint32_t key)
 | |
| {
 | |
|   for(uint16_t i=0; i<p_table->count; i++)
 | |
|   {
 | |
|     if (p_table->items[i].key == key) return p_table->items[i].data;
 | |
|   }
 | |
| 
 | |
|   return NULL;
 | |
| }
 | |
| 
 | |
| #endif // CFG_TUSB_DEBUG
 | |
| 
 | |
| #ifndef TU_LOG1
 | |
|   #define TU_LOG1(...)
 | |
|   #define TU_LOG1_MEM(...)
 | |
| #endif
 | |
| 
 | |
| #ifndef TU_LOG2
 | |
|   #define TU_LOG2(...)
 | |
|   #define TU_LOG2_MEM(...)
 | |
| #endif
 | |
| 
 | |
| #ifdef __cplusplus
 | |
|  }
 | |
| #endif
 | |
| 
 | |
| #endif /* _TUSB_COMMON_H_ */
 | |
| 
 | |
| /**  @} */
 | 
