/* * Copyright (c) 2010-2013 BitTorrent, Inc. * * 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. */ #ifndef __TEMPLATES_H__ #define __TEMPLATES_H__ #include "utp_types.h" #include #if defined(POSIX) /* Allow over-writing FORCEINLINE from makefile because gcc 3.4.4 for buffalo doesn't seem to support __attribute__((always_inline)) in -O0 build (strangely, it works in -Os build) */ #ifndef FORCEINLINE // The always_inline attribute asks gcc to inline the function even if no optimization is being requested. // This macro should be used exclusive-or with the inline directive (use one or the other but not both) // since Microsoft uses __forceinline to also mean inline, // and this code is following a Microsoft compatibility model. // Just setting the attribute without also specifying the inline directive apparently won't inline the function, // as evidenced by multiply-defined symbols found at link time. #define FORCEINLINE inline __attribute__((always_inline)) #endif #endif // Utility templates #undef min #undef max template static inline T min(T a, T b) { if (a < b) return a; return b; } template static inline T max(T a, T b) { if (a > b) return a; return b; } template static inline T min(T a, T b, T c) { return min(min(a,b),c); } template static inline T max(T a, T b, T c) { return max(max(a,b),c); } template static inline T clamp(T v, T mi, T ma) { if (v > ma) v = ma; if (v < mi) v = mi; return v; } #if (defined(__SVR4) && defined(__sun)) #pragma pack(1) #else #pragma pack(push,1) #endif namespace aux { FORCEINLINE uint16 host_to_network(uint16 i) { return htons(i); } FORCEINLINE uint32 host_to_network(uint32 i) { return htonl(i); } FORCEINLINE int32 host_to_network(int32 i) { return htonl(i); } FORCEINLINE uint16 network_to_host(uint16 i) { return ntohs(i); } FORCEINLINE uint32 network_to_host(uint32 i) { return ntohl(i); } FORCEINLINE int32 network_to_host(int32 i) { return ntohl(i); } } template struct PACKED_ATTRIBUTE big_endian { T operator=(T i) { m_integer = aux::host_to_network(i); return i; } operator T() const { return aux::network_to_host(m_integer); } private: T m_integer; }; typedef big_endian int32_big; typedef big_endian uint32_big; typedef big_endian uint16_big; #if (defined(__SVR4) && defined(__sun)) #pragma pack(0) #else #pragma pack(pop) #endif template static inline void zeromem(T *a, size_t count = 1) { memset(a, 0, count * sizeof(T)); } typedef int SortCompareProc(const void *, const void *); template static FORCEINLINE void QuickSortT(T *base, size_t num, int (*comp)(const T *, const T *)) { qsort(base, num, sizeof(T), (SortCompareProc*)comp); } // WARNING: The template parameter MUST be a POD type! template class Array { protected: T *mem; size_t alloc,count; public: Array(size_t init) { Init(init); } Array() { Init(); } ~Array() { Free(); } void inline Init() { mem = NULL; alloc = count = 0; } void inline Init(size_t init) { Init(); if (init) Resize(init); } size_t inline GetCount() const { return count; } size_t inline GetAlloc() const { return alloc; } void inline SetCount(size_t c) { count = c; } inline T& operator[](size_t offset) { assert(offset ==0 || offset(minsize, alloc * 2)); } inline size_t Append(const T &t) { if (count >= alloc) Grow(); size_t r=count++; mem[r] = t; return r; } T inline &Append() { if (count >= alloc) Grow(); return mem[count++]; } void inline Compact() { Resize(count); } void inline Free() { free(mem); Init(); } void inline Clear() { count = 0; } bool inline MoveUpLast(size_t index) { assert(index < count); size_t c = --count; if (index != c) { mem[index] = mem[c]; return true; } return false; } bool inline MoveUpLastExist(const T &v) { return MoveUpLast(LookupElementExist(v)); } size_t inline LookupElement(const T &v) const { for(size_t i = 0; i != count; i++) if (mem[i] == v) return i; return (size_t) -1; } bool inline HasElement(const T &v) const { return LookupElement(v) != -1; } typedef int SortCompareProc(const T *a, const T *b); void Sort(SortCompareProc* proc, size_t start, size_t end) { QuickSortT(&mem[start], end - start, proc); } void Sort(SortCompareProc* proc, size_t start) { Sort(proc, start, count); } void Sort(SortCompareProc* proc) { Sort(proc, 0, count); } }; #endif //__TEMPLATES_H__