2014-03-10 14:20:38 +07:00
|
|
|
/**************************************************************************/
|
|
|
|
/*!
|
2018-03-28 14:16:14 +07:00
|
|
|
@file tusb_common.h
|
2014-03-10 14:20:38 +07:00
|
|
|
@author hathach (tinyusb.org)
|
|
|
|
|
|
|
|
@section LICENSE
|
|
|
|
|
|
|
|
Software License Agreement (BSD License)
|
|
|
|
|
|
|
|
Copyright (c) 2013, hathach (tinyusb.org)
|
|
|
|
All rights reserved.
|
|
|
|
|
|
|
|
Redistribution and use in source and binary forms, with or without
|
|
|
|
modification, are permitted provided that the following conditions are met:
|
|
|
|
1. Redistributions of source code must retain the above copyright
|
|
|
|
notice, this list of conditions and the following disclaimer.
|
|
|
|
2. Redistributions in binary form must reproduce the above copyright
|
|
|
|
notice, this list of conditions and the following disclaimer in the
|
|
|
|
documentation and/or other materials provided with the distribution.
|
|
|
|
3. Neither the name of the copyright holders nor the
|
|
|
|
names of its contributors may be used to endorse or promote products
|
|
|
|
derived from this software without specific prior written permission.
|
|
|
|
|
|
|
|
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS ''AS IS'' AND ANY
|
|
|
|
EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
|
|
|
WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
|
|
|
DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER BE LIABLE FOR ANY
|
|
|
|
DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
|
|
|
INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
|
|
|
LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION HOWEVER CAUSED AND
|
|
|
|
ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
|
|
INCLUDING NEGLIGENCE OR OTHERWISE ARISING IN ANY WAY OUT OF THE USE OF THIS
|
|
|
|
SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
|
|
|
|
This file is part of the tinyusb stack.
|
|
|
|
*/
|
|
|
|
/**************************************************************************/
|
|
|
|
|
2014-03-25 13:11:22 +07:00
|
|
|
/** \ingroup Group_Common
|
2014-03-10 14:20:38 +07:00
|
|
|
* \defgroup Group_CommonH common.h
|
2014-03-16 22:20:33 +07:00
|
|
|
* @{ */
|
2014-03-10 14:20:38 +07:00
|
|
|
|
|
|
|
#ifndef _TUSB_COMMON_H_
|
|
|
|
#define _TUSB_COMMON_H_
|
|
|
|
|
|
|
|
#ifdef __cplusplus
|
|
|
|
extern "C" {
|
|
|
|
#endif
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------+
|
|
|
|
// INCLUDES
|
|
|
|
//--------------------------------------------------------------------+
|
|
|
|
|
|
|
|
//------------- Standard Header -------------//
|
2015-05-01 18:34:08 +07:00
|
|
|
#include <stdbool.h>
|
|
|
|
#include <stdint.h>
|
2014-03-10 14:20:38 +07:00
|
|
|
#include <stddef.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
|
|
|
|
//------------- TUSB Option Header -------------//
|
|
|
|
#include "tusb_option.h"
|
|
|
|
|
2018-07-23 23:41:14 +07:00
|
|
|
//------------- Common Header -------------//
|
2018-03-28 13:38:17 +07:00
|
|
|
#include "tusb_compiler.h"
|
2018-03-29 13:44:29 +07:00
|
|
|
#include "tusb_verify.h"
|
2014-03-10 14:20:38 +07:00
|
|
|
#include "binary.h"
|
2018-03-28 13:05:48 +07:00
|
|
|
#include "tusb_error.h"
|
2018-07-23 23:41:14 +07:00
|
|
|
#include "tusb_timeout.h"
|
2014-03-10 14:20:38 +07:00
|
|
|
#include "tusb_types.h"
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------+
|
|
|
|
// MACROS
|
|
|
|
//--------------------------------------------------------------------+
|
2018-10-23 12:19:32 +07:00
|
|
|
#define TU_ARRAY_SZIE(_arr) ( sizeof(_arr) / sizeof(_arr[0]) )
|
2018-12-10 19:01:28 +07:00
|
|
|
#define TU_MIN(_x, _y) ( (_x) < (_y) ) ? (_x) : (_y) )
|
|
|
|
#define TU_MAX(_x, _y) ( (_x) > (_y) ) ? (_x) : (_y) )
|
2014-03-10 14:20:38 +07:00
|
|
|
|
|
|
|
#define U16_HIGH_U8(u16) ((uint8_t) (((u16) >> 8) & 0x00ff))
|
|
|
|
#define U16_LOW_U8(u16) ((uint8_t) ((u16) & 0x00ff))
|
|
|
|
#define U16_TO_U8S_BE(u16) U16_HIGH_U8(u16), U16_LOW_U8(u16)
|
|
|
|
#define U16_TO_U8S_LE(u16) U16_LOW_U8(u16), U16_HIGH_U8(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)
|
|
|
|
|
|
|
|
//------------- Endian Conversion -------------//
|
|
|
|
#define ENDIAN_BE(u32) \
|
|
|
|
(uint32_t) ( (((u32) & 0xFF) << 24) | (((u32) & 0xFF00) << 8) | (((u32) >> 8) & 0xFF00) | (((u32) >> 24) & 0xFF) )
|
|
|
|
|
|
|
|
#define ENDIAN_BE16(le16) ((uint16_t) ((U16_LOW_U8(le16) << 8) | U16_HIGH_U8(le16)) )
|
|
|
|
|
|
|
|
#ifndef __n2be_16
|
|
|
|
#define __n2be_16(u16) ((uint16_t) ((U16_LOW_U8(u16) << 8) | U16_HIGH_U8(u16)) )
|
|
|
|
#define __be2n_16(u16) __n2be_16(u16)
|
|
|
|
#endif
|
|
|
|
|
2018-07-02 17:32:09 +07:00
|
|
|
|
2018-07-25 16:56:57 +07:00
|
|
|
// for declaration of reserved field, make use of _TU_COUNTER_
|
|
|
|
#define TU_RESERVED XSTRING_CONCAT_(reserved, _TU_COUNTER_)
|
|
|
|
|
2018-07-02 17:32:09 +07:00
|
|
|
/*------------------------------------------------------------------*/
|
|
|
|
/* 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 VA_ARGS_NUM_
|
|
|
|
|
|
|
|
#define VA_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
|
|
|
|
|
2014-03-10 14:20:38 +07:00
|
|
|
//--------------------------------------------------------------------+
|
|
|
|
// INLINE FUNCTION
|
|
|
|
//--------------------------------------------------------------------+
|
2018-10-23 12:19:32 +07:00
|
|
|
#define tu_memclr(buffer, size) memset((buffer), 0, (size))
|
|
|
|
#define tu_varclr(_var) tu_memclr(_var, sizeof(*(_var)))
|
2018-02-28 16:45:54 +07:00
|
|
|
|
2018-10-23 12:19:32 +07:00
|
|
|
static inline bool tu_mem_test_zero (void const* buffer, uint32_t size)
|
2014-03-10 14:20:38 +07:00
|
|
|
{
|
2018-07-24 18:17:09 +07:00
|
|
|
uint8_t const* p_mem = (uint8_t const*) buffer;
|
|
|
|
for(uint32_t i=0; i<size; i++) if (p_mem[i] != 0) return false;
|
|
|
|
return true;
|
2018-07-13 17:48:26 +07:00
|
|
|
}
|
|
|
|
|
2014-03-10 14:20:38 +07:00
|
|
|
|
|
|
|
//------------- Conversion -------------//
|
2018-08-23 20:09:28 +07:00
|
|
|
static inline uint32_t tu_u32_from_u8(uint8_t b1, uint8_t b2, uint8_t b3, uint8_t b4)
|
2014-03-10 14:20:38 +07:00
|
|
|
{
|
|
|
|
return ( ((uint32_t) b1) << 24) + ( ((uint32_t) b2) << 16) + ( ((uint32_t) b3) << 8) + b4;
|
|
|
|
}
|
|
|
|
|
2018-08-23 20:09:28 +07:00
|
|
|
static inline uint8_t tu_u16_high(uint16_t u16)
|
2014-03-10 14:20:38 +07:00
|
|
|
{
|
|
|
|
return (uint8_t) ( ((uint16_t) (u16 >> 8)) & 0x00ff);
|
|
|
|
}
|
|
|
|
|
2018-08-23 20:09:28 +07:00
|
|
|
static inline uint8_t tu_u16_low(uint16_t u16)
|
2014-03-10 14:20:38 +07:00
|
|
|
{
|
|
|
|
return (uint8_t) (u16 & 0x00ff);
|
|
|
|
}
|
|
|
|
|
2018-08-23 20:09:28 +07:00
|
|
|
static inline uint16_t tu_u16_le2be(uint16_t u16)
|
2014-03-10 14:20:38 +07:00
|
|
|
{
|
2018-08-23 20:09:28 +07:00
|
|
|
return ((uint16_t)(tu_u16_low(u16) << 8)) | tu_u16_high(u16);
|
2014-03-10 14:20:38 +07:00
|
|
|
}
|
|
|
|
|
2018-12-10 19:01:28 +07:00
|
|
|
// 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; }
|
2014-03-10 14:20:38 +07:00
|
|
|
|
2018-12-10 19:01:28 +07:00
|
|
|
// 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; }
|
2014-03-10 14:20:38 +07:00
|
|
|
|
2018-12-10 19:01:28 +07:00
|
|
|
// Align
|
|
|
|
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_align_n (uint32_t alignment, uint32_t value) { return value & ((uint32_t) ~(alignment-1)); }
|
|
|
|
static inline uint32_t tu_align4k (uint32_t value) { return (value & 0xFFFFF000UL); }
|
2014-03-10 14:20:38 +07:00
|
|
|
|
2018-12-10 19:01:28 +07:00
|
|
|
static inline uint32_t tu_offset4k(uint32_t value) { return (value & 0xFFFUL); }
|
2014-03-10 14:20:38 +07:00
|
|
|
|
|
|
|
//------------- Mathematics -------------//
|
2018-08-23 20:09:28 +07:00
|
|
|
static inline uint32_t tu_abs(int32_t value)
|
2014-03-10 14:20:38 +07:00
|
|
|
{
|
|
|
|
return (value < 0) ? (-value) : value;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// inclusive range checking
|
2018-08-23 20:09:28 +07:00
|
|
|
static inline bool tu_within(uint32_t lower, uint32_t value, uint32_t upper)
|
2014-03-10 14:20:38 +07:00
|
|
|
{
|
|
|
|
return (lower <= value) && (value <= upper);
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO use clz
|
2018-08-23 20:09:28 +07:00
|
|
|
static inline uint8_t tu_log2(uint32_t value)
|
2014-03-10 14:20:38 +07:00
|
|
|
{
|
|
|
|
uint8_t result = 0; // log2 of a value is its MSB's position
|
|
|
|
|
|
|
|
while (value >>= 1)
|
|
|
|
{
|
|
|
|
result++;
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef __cplusplus
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#endif /* _TUSB_COMMON_H_ */
|
|
|
|
|
|
|
|
/** @} */
|