| 
									
										
										
										
											2013-05-06 20:51:34 +07:00
										 |  |  | /**************************************************************************/ | 
					
						
							|  |  |  | /*!
 | 
					
						
							|  |  |  |     @file     common.h | 
					
						
							|  |  |  |     @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. | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | /**************************************************************************/ | 
					
						
							| 
									
										
										
										
											2012-11-29 10:41:33 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-03 16:51:14 +07:00
										 |  |  | /** \defgroup Group_Common Common Files
 | 
					
						
							|  |  |  |  * @{ | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *  \defgroup Group_CommonH common.h | 
					
						
							| 
									
										
										
										
											2012-11-29 10:41:33 +07:00
										 |  |  |  * | 
					
						
							|  |  |  |  *  @{ | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-27 15:21:47 +07:00
										 |  |  | #ifndef _TUSB_COMMON_H_
 | 
					
						
							|  |  |  | #define _TUSB_COMMON_H_
 | 
					
						
							| 
									
										
										
										
											2012-11-26 17:19:26 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-02 11:07:59 +07:00
										 |  |  | #ifdef __cplusplus
 | 
					
						
							|  |  |  |  extern "C" { | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-03 12:31:18 +07:00
										 |  |  | //--------------------------------------------------------------------+
 | 
					
						
							|  |  |  | // INCLUDES
 | 
					
						
							|  |  |  | //--------------------------------------------------------------------+
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-04 13:13:07 +07:00
										 |  |  | //------------- Standard Header -------------//
 | 
					
						
							| 
									
										
										
										
											2013-01-30 11:35:37 +07:00
										 |  |  | #include "primitive_types.h"
 | 
					
						
							| 
									
										
										
										
											2013-01-18 14:39:42 +07:00
										 |  |  | #include <stddef.h>
 | 
					
						
							| 
									
										
										
										
											2012-11-27 17:41:00 +07:00
										 |  |  | #include <string.h>
 | 
					
						
							| 
									
										
										
										
											2012-11-29 15:35:59 +07:00
										 |  |  | #include <stdio.h>
 | 
					
						
							| 
									
										
										
										
											2012-11-27 17:41:00 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-04 13:13:07 +07:00
										 |  |  | //------------- TUSB Option Header -------------//
 | 
					
						
							|  |  |  | #include "tusb_option.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | //------------- General Header -------------//
 | 
					
						
							| 
									
										
										
										
											2012-12-07 17:59:46 +07:00
										 |  |  | #include "compiler/compiler.h"
 | 
					
						
							| 
									
										
										
										
											2013-01-11 20:49:46 +07:00
										 |  |  | #include "assertion.h"
 | 
					
						
							|  |  |  | #include "binary.h"
 | 
					
						
							| 
									
										
										
										
											2013-01-31 11:47:07 +07:00
										 |  |  | #include "errors.h"
 | 
					
						
							| 
									
										
										
										
											2012-12-07 17:59:46 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-04 13:13:07 +07:00
										 |  |  | //------------- TUSB Header -------------//
 | 
					
						
							| 
									
										
										
										
											2012-12-03 16:51:14 +07:00
										 |  |  | #include "core/tusb_types.h"
 | 
					
						
							| 
									
										
										
										
											2012-12-04 11:30:34 +07:00
										 |  |  | #include "core/std_descriptors.h"
 | 
					
						
							| 
									
										
										
										
											2013-02-02 15:36:20 +07:00
										 |  |  | #include "core/std_request.h"
 | 
					
						
							| 
									
										
										
										
											2013-01-19 00:57:58 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-03 12:31:18 +07:00
										 |  |  | //--------------------------------------------------------------------+
 | 
					
						
							|  |  |  | // MACROS
 | 
					
						
							|  |  |  | //--------------------------------------------------------------------+
 | 
					
						
							| 
									
										
										
										
											2013-02-26 17:28:05 +07:00
										 |  |  | #define STRING_(x)  #x                             // stringify without expand
 | 
					
						
							|  |  |  | #define XSTRING_(x) STRING_(x)                     // expand then stringify
 | 
					
						
							|  |  |  | #define STRING_CONCAT_(a, b) a##b                  // concat without expand
 | 
					
						
							|  |  |  | #define XSTRING_CONCAT_(a, b) STRING_CONCAT_(a, b) // expand then concat
 | 
					
						
							| 
									
										
										
										
											2013-01-27 13:50:40 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-12 19:32:32 +07:00
										 |  |  | #define U16_HIGH_U8(u16) ((uint8_t) (((u16) >> 8) & 0x00ff))
 | 
					
						
							| 
									
										
										
										
											2013-02-05 13:57:06 +07:00
										 |  |  | #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)
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-12 19:32:32 +07:00
										 |  |  | #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))
 | 
					
						
							| 
									
										
										
										
											2013-02-05 13:57:06 +07:00
										 |  |  | #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)
 | 
					
						
							| 
									
										
										
										
											2013-02-06 12:03:01 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-03 12:31:18 +07:00
										 |  |  | //--------------------------------------------------------------------+
 | 
					
						
							|  |  |  | // INLINE FUNCTION
 | 
					
						
							|  |  |  | //--------------------------------------------------------------------+
 | 
					
						
							| 
									
										
										
										
											2013-02-06 12:03:01 +07:00
										 |  |  | #define memclr_(buffer, size)  memset(buffer, 0, size)
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-07-01 10:35:37 +07:00
										 |  |  | 
 | 
					
						
							|  |  |  | static inline uint8_t const * descriptor_next(uint8_t const * p_desc) ATTR_ALWAYS_INLINE ATTR_PURE; | 
					
						
							|  |  |  | static inline uint8_t const * descriptor_next(uint8_t const * p_desc) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   return p_desc + p_desc[DESCRIPTOR_OFFSET_LENGTH]; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline uint8_t descriptor_typeof(uint8_t const * p_desc) ATTR_ALWAYS_INLINE ATTR_PURE; | 
					
						
							|  |  |  | static inline uint8_t descriptor_typeof(uint8_t const * p_desc) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   return p_desc[DESCRIPTOR_OFFSET_TYPE]; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-03 12:31:18 +07:00
										 |  |  | //------------- Conversion -------------//
 | 
					
						
							| 
									
										
										
										
											2013-01-22 17:41:06 +07:00
										 |  |  | /// form an uint32_t from 4 x uint8_t
 | 
					
						
							|  |  |  | static inline uint32_t u32_from_u8(uint8_t b1, uint8_t b2, uint8_t b3, uint8_t b4) ATTR_ALWAYS_INLINE ATTR_CONST; | 
					
						
							|  |  |  | static inline uint32_t u32_from_u8(uint8_t b1, uint8_t b2, uint8_t b3, uint8_t b4) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   return (b1 << 24) + (b2 << 16) + (b3 << 8) + b4; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-12 19:32:32 +07:00
										 |  |  | static inline uint8_t u16_high_u8(uint16_t u16) ATTR_CONST ATTR_ALWAYS_INLINE; | 
					
						
							|  |  |  | static inline uint8_t u16_high_u8(uint16_t u16) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   return (uint8_t) ((u16 >> 8) & 0x00ff); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline uint8_t u16_low_u8(uint16_t u16) ATTR_CONST ATTR_ALWAYS_INLINE; | 
					
						
							|  |  |  | static inline uint8_t u16_low_u8(uint16_t u16) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   return (uint8_t) (u16 & 0x00ff); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-09-26 00:17:45 +07:00
										 |  |  | static inline uint16_t u16_le2be(uint16_t u16) ATTR_CONST ATTR_ALWAYS_INLINE; | 
					
						
							|  |  |  | static inline uint16_t u16_le2be(uint16_t u16) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   return (u16_low_u8(u16) << 8) | u16_high_u8(u16); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-03 12:31:18 +07:00
										 |  |  | //------------- Min -------------//
 | 
					
						
							| 
									
										
										
										
											2013-02-06 18:12:26 +07:00
										 |  |  | static inline uint8_t min8_of(uint8_t x, uint8_t y) ATTR_ALWAYS_INLINE ATTR_CONST; | 
					
						
							|  |  |  | static inline uint8_t min8_of(uint8_t x, uint8_t y) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   return (x < y) ? x : y; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-21 00:28:25 +07:00
										 |  |  | static inline uint16_t min16_of(uint16_t x, uint16_t y) ATTR_ALWAYS_INLINE ATTR_CONST; | 
					
						
							|  |  |  | static inline uint16_t min16_of(uint16_t x, uint16_t y) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   return (x < y) ? x : y; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-06 18:12:26 +07:00
										 |  |  | static inline uint32_t min32_of(uint32_t x, uint32_t y) ATTR_ALWAYS_INLINE ATTR_CONST; | 
					
						
							|  |  |  | static inline uint32_t min32_of(uint32_t x, uint32_t y) | 
					
						
							| 
									
										
										
										
											2013-01-14 00:41:23 +07:00
										 |  |  | { | 
					
						
							|  |  |  |   return (x < y) ? x : y; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2012-12-02 15:46:12 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-03 12:31:18 +07:00
										 |  |  | //------------- Max -------------//
 | 
					
						
							| 
									
										
										
										
											2013-02-06 18:12:26 +07:00
										 |  |  | static inline uint32_t max32_of(uint32_t x, uint32_t y) ATTR_ALWAYS_INLINE ATTR_CONST; | 
					
						
							|  |  |  | static inline uint32_t max32_of(uint32_t x, uint32_t y) | 
					
						
							| 
									
										
										
										
											2013-01-14 00:41:23 +07:00
										 |  |  | { | 
					
						
							|  |  |  |   return (x > y) ? x : y; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2012-12-02 15:46:12 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-03 12:31:18 +07:00
										 |  |  | //------------- Align -------------//
 | 
					
						
							|  |  |  | static inline uint32_t align32 (uint32_t value) ATTR_ALWAYS_INLINE ATTR_CONST; | 
					
						
							|  |  |  | static inline uint32_t align32 (uint32_t value) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return (value & 0xFFFFFFE0UL); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline uint32_t align16 (uint32_t value) ATTR_ALWAYS_INLINE ATTR_CONST; | 
					
						
							|  |  |  | static inline uint32_t align16 (uint32_t value) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return (value & 0xFFFFFFF0UL); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline uint32_t align_n (uint32_t alignment, uint32_t value) ATTR_ALWAYS_INLINE ATTR_CONST; | 
					
						
							|  |  |  | static inline uint32_t align_n (uint32_t alignment, uint32_t value) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return value & (~(alignment-1)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline uint32_t align4k (uint32_t value) ATTR_ALWAYS_INLINE ATTR_CONST; | 
					
						
							|  |  |  | static inline uint32_t align4k (uint32_t value) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return (value & 0xFFFFF000UL); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline uint32_t offset4k(uint32_t value) ATTR_ALWAYS_INLINE ATTR_CONST; | 
					
						
							|  |  |  | static inline uint32_t offset4k(uint32_t value) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return (value & 0xFFFUL); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-06 20:51:34 +07:00
										 |  |  | //------------- Mathematics -------------//
 | 
					
						
							| 
									
										
										
										
											2013-07-04 10:47:31 +07:00
										 |  |  | /// inclusive range checking
 | 
					
						
							|  |  |  | static inline bool is_in_range(uint32_t lower, uint32_t value, uint32_t upper) ATTR_ALWAYS_INLINE ATTR_CONST; | 
					
						
							|  |  |  | static inline bool is_in_range(uint32_t lower, uint32_t value, uint32_t upper) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   return (lower <= value) && (value <= upper); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /// exclusive range checking
 | 
					
						
							|  |  |  | static inline bool is_in_range_exclusive(uint32_t lower, uint32_t value, uint32_t upper) ATTR_ALWAYS_INLINE ATTR_CONST; | 
					
						
							|  |  |  | static inline bool is_in_range_exclusive(uint32_t lower, uint32_t value, uint32_t upper) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   return (lower < value) && (value < upper); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-06 20:51:34 +07:00
										 |  |  | static inline uint8_t log2_of(uint32_t value) ATTR_ALWAYS_INLINE ATTR_CONST; | 
					
						
							|  |  |  | static inline uint8_t log2_of(uint32_t value) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-05-11 00:40:21 +07:00
										 |  |  |   uint8_t result = 0; // log2 of a value is its MSB's position
 | 
					
						
							| 
									
										
										
										
											2013-05-06 20:51:34 +07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   while (value >>= 1) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     result++; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return result; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-11 00:40:21 +07:00
										 |  |  | // return the number of set bits in value
 | 
					
						
							|  |  |  | static inline uint8_t cardinality_of(uint32_t value) ATTR_ALWAYS_INLINE ATTR_CONST; | 
					
						
							|  |  |  | static inline uint8_t cardinality_of(uint32_t value) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   // Brian Kernighan's method goes through as many iterations as there are set bits. So if we have a 32-bit word with only
 | 
					
						
							|  |  |  |   // the high bit set, then it will only go once through the loop
 | 
					
						
							|  |  |  |   // Published in 1988, the C Programming Language 2nd Ed. (by Brian W. Kernighan and Dennis M. Ritchie)
 | 
					
						
							|  |  |  |   // mentions this in exercise 2-9. On April 19, 2006 Don Knuth pointed out to me that this method
 | 
					
						
							|  |  |  |   // "was first published by Peter Wegner in CACM 3 (1960), 322. (Also discovered independently by Derrick Lehmer and
 | 
					
						
							|  |  |  |   // published in 1964 in a book edited by Beckenbach.)"
 | 
					
						
							|  |  |  |   uint8_t count; | 
					
						
							|  |  |  |   for (count = 0; value; count++) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     value &= value - 1; // clear the least significant bit set
 | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return count; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2013-05-06 20:51:34 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-02 11:07:59 +07:00
										 |  |  | #ifdef __cplusplus
 | 
					
						
							|  |  |  |  } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-27 15:21:47 +07:00
										 |  |  | #endif /* _TUSB_COMMON_H_ */
 | 
					
						
							| 
									
										
										
										
											2012-11-29 15:35:59 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-29 15:40:57 +07:00
										 |  |  | /**  @} */ | 
					
						
							| 
									
										
										
										
											2012-12-03 16:51:14 +07:00
										 |  |  | /**  @} */ |