root/usr/src/compat/bhyve/sys/endian.h
/*
 * This file and its contents are supplied under the terms of the
 * Common Development and Distribution License ("CDDL"), version 1.0.
 * You may only use this file in accordance with the terms of version
 * 1.0 of the CDDL.
 *
 * A full copy of the text of the CDDL should have accompanied this
 * source.  A copy of the CDDL is also available via the Internet at
 * http://www.illumos.org/license/CDDL.
 */

/*
 * Copyright 2014 Pluribus Networks Inc.
 * Copyright 2018 Joyent, Inc.
 */

#ifndef _COMPAT_FREEBSD_SYS_ENDIAN_H_
#define _COMPAT_FREEBSD_SYS_ENDIAN_H_

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 uint64_t
be64dec(const void *pp)
{
        uint8_t const *p = (uint8_t const *)pp;

        return (((uint64_t)be32dec(p) << 32) | be32dec(p + 4));
}

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 (((unsigned)p[3] << 24) | (p[2] << 16) | (p[1] << 8) | p[0]);
}

static __inline uint64_t
le64dec(const void *pp)
{
        uint8_t const *p = (uint8_t const *)pp;

        return (((uint64_t)le32dec(p + 4) << 32) | le32dec(p));
}

static __inline void
be16enc(void *pp, uint16_t u)
{
        uint8_t *p = (uint8_t *)pp;

        p[0] = (u >> 8) & 0xff;
        p[1] = u & 0xff;
}

static __inline void
be32enc(void *pp, uint32_t u)
{
        uint8_t *p = (uint8_t *)pp;

        p[0] = (u >> 24) & 0xff;
        p[1] = (u >> 16) & 0xff;
        p[2] = (u >> 8) & 0xff;
        p[3] = u & 0xff;
}

static __inline void
be64enc(void *pp, uint64_t u)
{
        uint8_t *p = (uint8_t *)pp;

        be32enc(p, (uint32_t)(u >> 32));
        be32enc(p + 4, (uint32_t)(u & 0xffffffffU));
}

static __inline void
le16enc(void *pp, uint16_t u)
{
        uint8_t *p = (uint8_t *)pp;

        p[0] = u & 0xff;
        p[1] = (u >> 8) & 0xff;
}

static __inline void
le32enc(void *pp, uint32_t u)
{
        uint8_t *p = (uint8_t *)pp;

        p[0] = u & 0xff;
        p[1] = (u >> 8) & 0xff;
        p[2] = (u >> 16) & 0xff;
        p[3] = (u >> 24) & 0xff;
}

static __inline void
le64enc(void *pp, uint64_t u)
{
        uint8_t *p = (uint8_t *)pp;

        le32enc(p, (uint32_t)(u & 0xffffffffU));
        le32enc(p + 4, (uint32_t)(u >> 32));
}

#ifdef _LITTLE_ENDIAN
#define htole16(x)      ((uint16_t)(x))
#define htole32(x)      ((uint32_t)(x))
#define htole64(x)      ((uint64_t)(x))

#define le16toh(x)      ((uint16_t)(x))
#define le32toh(x)      ((uint32_t)(x))
#define le64toh(x)      ((uint64_t)(x))
#endif

#endif  /* _COMPAT_FREEBSD_SYS_ENDIAN_H_ */