ByteReverse.cxx 3.06 KB
Newer Older
1
/*
2
 * Copyright 2003-2018 The Music Player Daemon Project
3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19
 * http://www.musicpd.org
 *
 * 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.,
 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
 */

20
#include "ByteReverse.hxx"
21
#include "system/ByteOrder.hxx"
22
#include "Compiler.h"
23 24 25 26

#include <assert.h>

void
27
reverse_bytes_16(uint16_t *gcc_restrict dest,
28 29
		 const uint16_t *gcc_restrict src,
		 const uint16_t *src_end) noexcept
30
{
31 32
	assert(dest != nullptr);
	assert(src != nullptr);
33 34 35 36
	assert(src_end >= src);

	while (src < src_end) {
		const uint16_t x = *src++;
37
		*dest++ = ByteSwap16(x);
38 39 40 41
	}
}

void
42
reverse_bytes_32(uint32_t *gcc_restrict dest,
43 44
		 const uint32_t *gcc_restrict src,
		 const uint32_t *src_end) noexcept
45
{
46 47
	assert(dest != nullptr);
	assert(src != nullptr);
48 49 50 51
	assert(src_end >= src);

	while (src < src_end) {
		const uint32_t x = *src++;
52
		*dest++ = ByteSwap32(x);
53 54 55 56
	}
}

void
57
reverse_bytes_64(uint64_t *gcc_restrict dest,
58 59
		 const uint64_t *gcc_restrict src,
		 const uint64_t *src_end) noexcept
60
{
61 62
	assert(dest != nullptr);
	assert(src != nullptr);
63 64 65 66
	assert(src_end >= src);

	while (src < src_end) {
		const uint64_t x = *src++;
67
		*dest++ = ByteSwap64(x);
68 69 70 71
	}
}

static void
72
reverse_bytes_linear(uint8_t *gcc_restrict dest,
73
		     const uint8_t *gcc_restrict src, size_t n) noexcept
74 75 76 77 78 79 80 81
{
	src += n;

	while (n-- > 0)
		*dest++ = *--src;
}

static void
82 83
reverse_bytes_generic(uint8_t *gcc_restrict dest,
		      const uint8_t *gcc_restrict src, const uint8_t *src_end,
84
		      size_t frame_size) noexcept
85
{
86 87
	assert(dest != nullptr);
	assert(src != nullptr);
88 89 90 91 92 93 94 95 96 97 98 99
	assert(src_end >= src);
	assert(frame_size > 0);
	assert((src_end - src) % frame_size == 0);

	while (src < src_end) {
		reverse_bytes_linear(dest, src, frame_size);
		dest += frame_size;
		src += frame_size;
	}
}

void
100 101
reverse_bytes(uint8_t *gcc_restrict dest,
	      const uint8_t *gcc_restrict src, const uint8_t *src_end,
102
	      size_t frame_size) noexcept
103
{
104 105
	assert(dest != nullptr);
	assert(src != nullptr);
106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132
	assert(src_end >= src);
	assert(frame_size > 0);
	assert((src_end - src) % frame_size == 0);

	switch (frame_size) {
	case 2:
		reverse_bytes_16((uint16_t *)dest,
				 (const uint16_t *)src,
				 (const uint16_t *)src_end);
		break;

	case 4:
		reverse_bytes_32((uint32_t *)dest,
				 (const uint32_t *)src,
				 (const uint32_t *)src_end);
		break;

	case 8:
		reverse_bytes_64((uint64_t *)dest,
				 (const uint64_t *)src,
				 (const uint64_t *)src_end);
		break;

	default:
		reverse_bytes_generic(dest, src, src_end, frame_size);
	}
}