/* * Copyright (C) 2008 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ /* * Some handy functions for manipulating bits and bytes. * * These get inlined, so prefer small size over maximum speed. */ #ifndef _DALVIK_BITS #define _DALVIK_BITS #include "Common.h" #include "Inlines.h" #include <stdlib.h> #include <string.h> /* * Get 1 byte. (Included to make the code more legible.) */ INLINE u1 get1(unsigned const char* pSrc) { return *pSrc; } /* * Get 2 big-endian bytes. */ INLINE u2 get2BE(unsigned char const* pSrc) { return (pSrc[0] << 8) | pSrc[1]; } /* * Get 4 big-endian bytes. */ INLINE u4 get4BE(unsigned char const* pSrc) { return (pSrc[0] << 24) | (pSrc[1] << 16) | (pSrc[2] << 8) | pSrc[3]; } /* * Get 8 big-endian bytes. */ INLINE u8 get8BE(unsigned char const* pSrc) { u4 low, high; high = pSrc[0]; high = (high << 8) | pSrc[1]; high = (high << 8) | pSrc[2]; high = (high << 8) | pSrc[3]; low = pSrc[4]; low = (low << 8) | pSrc[5]; low = (low << 8) | pSrc[6]; low = (low << 8) | pSrc[7]; return ((u8) high << 32) | (u8) low; } /* * Get 2 little-endian bytes. */ INLINE u2 get2LE(unsigned char const* pSrc) { return pSrc[0] | (pSrc[1] << 8); } /* * Get 4 little-endian bytes. */ INLINE u4 get4LE(unsigned char const* pSrc) { u4 result; result = pSrc[0]; result |= pSrc[1] << 8; result |= pSrc[2] << 16; result |= pSrc[3] << 24; return result; } /* * Get 8 little-endian bytes. */ INLINE u8 get8LE(unsigned char const* pSrc) { u4 low, high; low = pSrc[0]; low |= pSrc[1] << 8; low |= pSrc[2] << 16; low |= pSrc[3] << 24; high = pSrc[4]; high |= pSrc[5] << 8; high |= pSrc[6] << 16; high |= pSrc[7] << 24; return ((u8) high << 32) | (u8) low; } /* * Grab 1 byte and advance the data pointer. */ INLINE u1 read1(unsigned const char** ppSrc) { return *(*ppSrc)++; } /* * Grab 2 big-endian bytes and advance the data pointer. */ INLINE u2 read2BE(unsigned char const** ppSrc) { const unsigned char* pSrc = *ppSrc; *ppSrc = pSrc + 2; return pSrc[0] << 8 | pSrc[1]; } /* * Grab 4 big-endian bytes and advance the data pointer. */ INLINE u4 read4BE(unsigned char const** ppSrc) { const unsigned char* pSrc = *ppSrc; u4 result; result = pSrc[0] << 24; result |= pSrc[1] << 16; result |= pSrc[2] << 8; result |= pSrc[3]; *ppSrc = pSrc + 4; return result; } /* * Get 8 big-endian bytes and advance the data pointer. */ INLINE u8 read8BE(unsigned char const** ppSrc) { const unsigned char* pSrc = *ppSrc; u4 low, high; high = pSrc[0]; high = (high << 8) | pSrc[1]; high = (high << 8) | pSrc[2]; high = (high << 8) | pSrc[3]; low = pSrc[4]; low = (low << 8) | pSrc[5]; low = (low << 8) | pSrc[6]; low = (low << 8) | pSrc[7]; *ppSrc = pSrc + 8; return ((u8) high << 32) | (u8) low; } /* * Grab 2 little-endian bytes and advance the data pointer. */ INLINE u2 read2LE(unsigned char const** ppSrc) { const unsigned char* pSrc = *ppSrc; *ppSrc = pSrc + 2; return pSrc[0] | pSrc[1] << 8; } /* * Grab 4 little-endian bytes and advance the data pointer. */ INLINE u4 read4LE(unsigned char const** ppSrc) { const unsigned char* pSrc = *ppSrc; u4 result; result = pSrc[0]; result |= pSrc[1] << 8; result |= pSrc[2] << 16; result |= pSrc[3] << 24; *ppSrc = pSrc + 4; return result; } /* * Get 8 little-endian bytes and advance the data pointer. */ INLINE u8 read8LE(unsigned char const** ppSrc) { const unsigned char* pSrc = *ppSrc; u4 low, high; low = pSrc[0]; low |= pSrc[1] << 8; low |= pSrc[2] << 16; low |= pSrc[3] << 24; high = pSrc[4]; high |= pSrc[5] << 8; high |= pSrc[6] << 16; high |= pSrc[7] << 24; *ppSrc = pSrc + 8; return ((u8) high << 32) | (u8) low; } /* * Skip over a UTF-8 string (preceded by a 4-byte length). */ INLINE void skipUtf8String(unsigned char const** ppSrc) { u4 length = read4BE(ppSrc); (*ppSrc) += length; } /* * Read a UTF-8 string into a fixed-size buffer, and null-terminate it. * * Returns the length of the original string. */ INLINE int readUtf8String(unsigned char const** ppSrc, char* buf, size_t bufLen) { u4 length = read4BE(ppSrc); size_t copyLen = (length < bufLen) ? length : bufLen-1; memcpy(buf, *ppSrc, copyLen); buf[copyLen] = '\0'; (*ppSrc) += length; return length; } /* * Read a UTF-8 string into newly-allocated storage, and null-terminate it. * * Returns the string and its length. (The latter is probably unnecessary * for the way we're using UTF8.) */ INLINE char* readNewUtf8String(unsigned char const** ppSrc, size_t* pLength) { u4 length = read4BE(ppSrc); char* buf; buf = (char*) malloc(length+1); memcpy(buf, *ppSrc, length); buf[length] = '\0'; (*ppSrc) += length; *pLength = length; return buf; } /* * Set 1 byte. (Included to make code more consistent/legible.) */ INLINE void set1(u1* buf, u1 val) { *buf = (u1)(val); } /* * Set 2 big-endian bytes. */ INLINE void set2BE(u1* buf, u2 val) { *buf++ = (u1)(val >> 8); *buf = (u1)(val); } /* * Set 4 big-endian bytes. */ INLINE void set4BE(u1* buf, u4 val) { *buf++ = (u1)(val >> 24); *buf++ = (u1)(val >> 16); *buf++ = (u1)(val >> 8); *buf = (u1)(val); } /* * Set 8 big-endian bytes. */ INLINE void set8BE(u1* buf, u8 val) { *buf++ = (u1)(val >> 56); *buf++ = (u1)(val >> 48); *buf++ = (u1)(val >> 40); *buf++ = (u1)(val >> 32); *buf++ = (u1)(val >> 24); *buf++ = (u1)(val >> 16); *buf++ = (u1)(val >> 8); *buf = (u1)(val); } /* * Set 2 little-endian bytes. */ INLINE void set2LE(u1* buf, u2 val) { *buf++ = (u1)(val); *buf = (u1)(val >> 8); } /* * Set 4 little-endian bytes. */ INLINE void set4LE(u1* buf, u4 val) { *buf++ = (u1)(val); *buf++ = (u1)(val >> 8); *buf++ = (u1)(val >> 16); *buf = (u1)(val >> 24); } /* * Set 8 little-endian bytes. */ INLINE void set8LE(u1* buf, u8 val) { *buf++ = (u1)(val); *buf++ = (u1)(val >> 8); *buf++ = (u1)(val >> 16); *buf++ = (u1)(val >> 24); *buf++ = (u1)(val >> 32); *buf++ = (u1)(val >> 40); *buf++ = (u1)(val >> 48); *buf = (u1)(val >> 56); } /* * Stuff a UTF-8 string into the buffer. */ INLINE void setUtf8String(u1* buf, const u1* str) { u4 strLen = strlen((const char*)str); set4BE(buf, strLen); memcpy(buf + sizeof(u4), str, strLen); } #endif /*_DALVIK_BITS*/