Alignment-agnostic encode/decode bytestream to/from little/big endian. The le16enc(), le16dec(), le32enc(), le32dec() functions encode and decode integers to/from byte strings on any alignment in big/little endian format. See BYTEORDER(9). Change-Id: I73a174b9c02c467bc60590c5cd894dac58b8683a Signed-off-by: Edward O'Callaghan <eocallaghan@alterapraxis.com> Reviewed-on: http://review.coreboot.org/5198 Tested-by: build bot (Jenkins) Reviewed-by: Ronald G. Minnich <rminnich@gmail.com>
		
			
				
	
	
		
			175 lines
		
	
	
		
			4.0 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			175 lines
		
	
	
		
			4.0 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/*
 | 
						|
 * Copyright (c) 2012 The Chromium OS Authors.
 | 
						|
 *
 | 
						|
 * See file CREDITS for list of people who contributed to this
 | 
						|
 * project.
 | 
						|
 *
 | 
						|
 * This program is free software; you can redistribute it and/or
 | 
						|
 * modify it under the terms of the GNU General Public License as
 | 
						|
 * published by the Free Software Foundation; either version 2 of
 | 
						|
 * the License, or (at your option) any later version.
 | 
						|
 *
 | 
						|
 * This program is distributed in the hope that it will be useful,
 | 
						|
 * but without any warranty; without even the implied warranty of
 | 
						|
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | 
						|
 * GNU General Public License for more details.
 | 
						|
 *
 | 
						|
 * You should have received a copy of the GNU General Public License
 | 
						|
 * along with this program; if not, write to the Free Software
 | 
						|
 * Foundation, Inc., 59 Temple Place, Suite 330, Boston,
 | 
						|
 * MA 02111-1307 USA
 | 
						|
 */
 | 
						|
 | 
						|
#ifndef _ENDIAN_H_
 | 
						|
#define _ENDIAN_H_
 | 
						|
 | 
						|
#include <arch/types.h>
 | 
						|
#include <libpayload-config.h>
 | 
						|
 | 
						|
static inline uint16_t swap_bytes16(uint16_t in)
 | 
						|
{
 | 
						|
	return ((in & 0xFF) << 8) | ((in & 0xFF00) >> 8);
 | 
						|
}
 | 
						|
 | 
						|
static inline uint32_t swap_bytes32(uint32_t in)
 | 
						|
{
 | 
						|
	return ((in & 0xFF) << 24) | ((in & 0xFF00) << 8) |
 | 
						|
		((in & 0xFF0000) >> 8) | ((in & 0xFF000000) >> 24);
 | 
						|
}
 | 
						|
 | 
						|
static inline uint64_t swap_bytes64(uint64_t in)
 | 
						|
{
 | 
						|
	return ((uint64_t)swap_bytes32((uint32_t)in) << 32) |
 | 
						|
		((uint64_t)swap_bytes32((uint32_t)(in >> 32)));
 | 
						|
}
 | 
						|
 | 
						|
/* Endian functions from glibc 2.9 / BSD "endian.h" */
 | 
						|
 | 
						|
#if defined CONFIG_BIG_ENDIAN
 | 
						|
 | 
						|
#define htobe16(in) (in)
 | 
						|
#define htobe32(in) (in)
 | 
						|
#define htobe64(in) (in)
 | 
						|
 | 
						|
#define htole16(in) swap_bytes16(in)
 | 
						|
#define htole32(in) swap_bytes32(in)
 | 
						|
#define htole64(in) swap_bytes64(in)
 | 
						|
 | 
						|
#elif defined CONFIG_LITTLE_ENDIAN
 | 
						|
 | 
						|
#define htobe16(in) swap_bytes16(in)
 | 
						|
#define htobe32(in) swap_bytes32(in)
 | 
						|
#define htobe64(in) swap_bytes64(in)
 | 
						|
 | 
						|
#define htole16(in) (in)
 | 
						|
#define htole32(in) (in)
 | 
						|
#define htole64(in) (in)
 | 
						|
 | 
						|
#else
 | 
						|
 | 
						|
#error Cant tell if the CPU is little or big endian.
 | 
						|
 | 
						|
#endif /* CONFIG_*_ENDIAN */
 | 
						|
 | 
						|
#define be16toh(in) htobe16(in)
 | 
						|
#define be32toh(in) htobe32(in)
 | 
						|
#define be64toh(in) htobe64(in)
 | 
						|
 | 
						|
#define le16toh(in) htole16(in)
 | 
						|
#define le32toh(in) htole32(in)
 | 
						|
#define le64toh(in) htole64(in)
 | 
						|
 | 
						|
#define htonw(in) htobe16(in)
 | 
						|
#define htonl(in) htobe32(in)
 | 
						|
#define htonll(in) htobe64(in)
 | 
						|
 | 
						|
#define ntohw(in) be16toh(in)
 | 
						|
#define ntohl(in) be32toh(in)
 | 
						|
#define ntohll(in) be64toh(in)
 | 
						|
 | 
						|
/*
 | 
						|
 * Alignment-agnostic encode/decode bytestream to/from little/big endian.
 | 
						|
 */
 | 
						|
 | 
						|
static inline uint16_t be16dec(const void *pp)
 | 
						|
{
 | 
						|
	uint8_t const *p = (uint8_t const *)pp;
 | 
						|
 | 
						|
	return ((p[0] << 8) | p[1]);
 | 
						|
}
 | 
						|
 | 
						|
static inline uint32_t be32dec(const void *pp)
 | 
						|
{
 | 
						|
	uint8_t const *p = (uint8_t const *)pp;
 | 
						|
 | 
						|
	return (((unsigned)p[0] << 24) | (p[1] << 16) | (p[2] << 8) | p[3]);
 | 
						|
}
 | 
						|
 | 
						|
static inline uint16_t le16dec(const void *pp)
 | 
						|
{
 | 
						|
	uint8_t const *p = (uint8_t const *)pp;
 | 
						|
 | 
						|
	return ((p[1] << 8) | p[0]);
 | 
						|
}
 | 
						|
 | 
						|
static inline uint32_t le32dec(const void *pp)
 | 
						|
{
 | 
						|
	uint8_t const *p = (uint8_t const *)pp;
 | 
						|
 | 
						|
	return ((p[3] << 24) | (p[2] << 16) | (p[1] << 8) | p[0]);
 | 
						|
}
 | 
						|
 | 
						|
static inline void bebitenc(void *pp, uint32_t u, uint8_t b)
 | 
						|
{
 | 
						|
	uint8_t *p = (uint8_t *)pp;
 | 
						|
	int i;
 | 
						|
 | 
						|
	for (i = (b - 1); i >= 0; i++)
 | 
						|
		p[i] = (u >> i*8) & 0xFF;
 | 
						|
}
 | 
						|
 | 
						|
static inline void be16enc(void *pp, uint16_t u)
 | 
						|
{
 | 
						|
	bebitenc(pp, u, 2);
 | 
						|
}
 | 
						|
 | 
						|
static inline void be32enc(void *pp, uint32_t u)
 | 
						|
{
 | 
						|
	bebitenc(pp, u, 4);
 | 
						|
}
 | 
						|
 | 
						|
static inline void lebitenc(void *pp, uint32_t u, uint8_t b)
 | 
						|
{
 | 
						|
	uint8_t *p = (uint8_t *)pp;
 | 
						|
	int i;
 | 
						|
 | 
						|
	for (i = 0; i < b; i++)
 | 
						|
		p[i] = (u >> i*8) & 0xFF;
 | 
						|
}
 | 
						|
 | 
						|
static inline void le16enc(void *pp, uint16_t u)
 | 
						|
{
 | 
						|
	lebitenc(pp, u, 2);
 | 
						|
}
 | 
						|
 | 
						|
static inline void le32enc(void *pp, uint32_t u)
 | 
						|
{
 | 
						|
	lebitenc(pp, u, 4);
 | 
						|
}
 | 
						|
 | 
						|
/* Deprecated names (not in glibc / BSD) */
 | 
						|
#define htobew(in) htobe16(in)
 | 
						|
#define htobel(in) htobe32(in)
 | 
						|
#define htobell(in) htobe64(in)
 | 
						|
#define htolew(in) htole16(in)
 | 
						|
#define htolel(in) htole32(in)
 | 
						|
#define htolell(in) htole64(in)
 | 
						|
#define betohw(in) be16toh(in)
 | 
						|
#define betohl(in) be32toh(in)
 | 
						|
#define betohll(in) be64toh(in)
 | 
						|
#define letohw(in) le16toh(in)
 | 
						|
#define letohl(in) le32toh(in)
 | 
						|
#define letohll(in) le64toh(in)
 | 
						|
 | 
						|
#endif /* _ENDIAN_H_ */
 |