aboutsummaryrefslogblamecommitdiffstats
path: root/rawmemelemcpy.c
blob: d5e52a22de4aef2d9b2f2658a494959430252b71 (plain) (tree)





































                                                                                
                                                                                                 

                 

                                           
                                    
                                           

                                            
                           

                         
                           




         
                                                                                                          
 
                        










                                                                      
                                                         






















































































                                                                                                                       
/* See LICENSE file for copyright and license details. */
#include "libsimple.h"
#ifndef TEST


static inline uint8_t *
rawmemelemcpy8(uint8_t *restrict d, const uint8_t *restrict s, uint16_t elem)
{
	for (; (*d++ = *s) != elem; s++);
	return d;
}


static inline uint16_t *
rawmemelemcpy16(uint16_t *restrict d, const uint16_t *restrict s, uint16_t elem)
{
	for (; (*d++ = *s) != elem; s++);
	return d;
}


static inline uint32_t *
rawmemelemcpy32(uint32_t *restrict d, const uint32_t *restrict s, uint32_t elem)
{
	for (; (*d++ = *s) != elem; s++);
	return d;
}


static inline uint64_t *
rawmemelemcpy64(uint64_t *restrict d, const uint64_t *restrict s, uint64_t elem)
{
	for (; (*d++ = *s) != elem; s++);
	return d;
}


static inline char *
rawmemelemcpyx(char *restrict d, const char *restrict s, const char *restrict elem, size_t width)
{
	size_t i;
	for (;; s += width) {
		for (i = 0; i < width; i++)
			d[i] = s[i];
		for (i = 0; i < width; i++)
			if (d[i] != elem[i])
				goto next;
		d += width;
		return d;
	next:
		d += width;
	}
}


void *
libsimple_rawmemelemcpy(void *restrict d, const void *restrict s, const void *restrict elem, size_t width)
{
	switch (width) {
	case 0:
		return d;
	case 1:
		return rawmemelemcpy8(d, s, *(const uint8_t *)elem);
	case 2:
		return rawmemelemcpy16(d, s, *(const uint16_t *)elem);
	case 4:
		return rawmemelemcpy32(d, s, *(const uint32_t *)elem);
	case 8:
		return rawmemelemcpy64(d, s, *(const uint64_t *)elem);
	default:
		return rawmemelemcpyx(d, s, elem, width);
	}
}


#else
#include "test.h"

int
main(void)
{
	char buf[1024];


	memset(buf, 'x', sizeof(buf)), buf[sizeof(buf) - 1] = '\0';
	assert(libsimple_rawmemelemcpy(buf, "hello", "o", 0) == &buf[0]);
	assert(!strncmp(buf, "x", 1));

	memset(buf, 'x', sizeof(buf)), buf[sizeof(buf) - 1] = '\0';
	assert(libsimple_rawmemelemcpy(buf, "hello", "l", 0) == &buf[0]);
	assert(!strncmp(buf, "x", 1));


	memset(buf, 'x', sizeof(buf)), buf[sizeof(buf) - 1] = '\0';
	assert(libsimple_rawmemelemcpy(buf, "hello", "o", 1) == &buf[5]);
	assert(!strncmp(buf, "hellox", 6));

	memset(buf, 'x', sizeof(buf)), buf[sizeof(buf) - 1] = '\0';
	assert(libsimple_rawmemelemcpy(buf, "hello", "l", 1) == &buf[3]);
	assert(!strncmp(buf, "helx", 4));


	memset(buf, 'x', sizeof(buf)), buf[sizeof(buf) - 1] = '\0';
	assert(libsimple_rawmemelemcpy(buf, ".h.e.l.l.o", ".o", 2) == &buf[5 * 2]);
	assert(!strncmp(buf, ".h.e.l.l.ox", 11));

	memset(buf, 'x', sizeof(buf)), buf[sizeof(buf) - 1] = '\0';
	assert(libsimple_rawmemelemcpy(buf, ".h.e.l.l.o", ".l", 2) == &buf[3 * 2]);
	assert(!strncmp(buf, ".h.e.lx", 7));

	memset(buf, 'x', sizeof(buf)), buf[sizeof(buf) - 1] = '\0';
	assert(libsimple_rawmemelemcpy(buf, ".h.l.ll..o", "l.", 2) == &buf[4 * 2]);
	assert(!strncmp(buf, ".h.l.ll.x", 9));


	memset(buf, 'x', sizeof(buf)), buf[sizeof(buf) - 1] = '\0';
	assert(libsimple_rawmemelemcpy(buf, "..h..e..l..l..o", "..o", 3) == &buf[5 * 3]);
	assert(!strncmp(buf, "..h..e..l..l..ox", 16));

	memset(buf, 'x', sizeof(buf)), buf[sizeof(buf) - 1] = '\0';
	assert(libsimple_rawmemelemcpy(buf, "..h..e..l..l..o", "..l", 3) == &buf[3 * 3]);
	assert(!strncmp(buf, "..h..e..lx", 10));

	memset(buf, 'x', sizeof(buf)), buf[sizeof(buf) - 1] = '\0';
	assert(libsimple_rawmemelemcpy(buf, "..h..l..ll....o", "l..", 3) == &buf[4 * 3]);
	assert(!strncmp(buf, "..h..l..ll..x", 13));


	memset(buf, 'x', sizeof(buf)), buf[sizeof(buf) - 1] = '\0';
	assert(libsimple_rawmemelemcpy(buf, "...h...e...l...l...o", "...o", 4) == &buf[5 * 4]);
	assert(!strncmp(buf, "...h...e...l...l...ox", 21));

	memset(buf, 'x', sizeof(buf)), buf[sizeof(buf) - 1] = '\0';
	assert(libsimple_rawmemelemcpy(buf, "...h...e...l...l...o", "...l", 4) == &buf[3 * 4]);
	assert(!strncmp(buf, "...h...e...lx", 13));

	memset(buf, 'x', sizeof(buf)), buf[sizeof(buf) - 1] = '\0';
	assert(libsimple_rawmemelemcpy(buf, "...h...l...ll......o", "l...", 4) == &buf[4 * 4]);
	assert(!strncmp(buf, "...h...l...ll...x", 17));


	memset(buf, 'x', sizeof(buf)), buf[sizeof(buf) - 1] = '\0';
	assert(libsimple_rawmemelemcpy(buf, ".......h.......e.......l.......l.......o", ".......o", 8) == &buf[5 * 8]);
	assert(!strncmp(buf, ".......h.......e.......l.......l.......ox", 21));

	memset(buf, 'x', sizeof(buf)), buf[sizeof(buf) - 1] = '\0';
	assert(libsimple_rawmemelemcpy(buf, ".......h.......e.......l.......l.......o", ".......l", 8) == &buf[3 * 8]);
	assert(!strncmp(buf, ".......h.......e.......lx", 13));

	memset(buf, 'x', sizeof(buf)), buf[sizeof(buf) - 1] = '\0';
	assert(libsimple_rawmemelemcpy(buf, ".......h.......l.......ll..............o", "l.......", 8) == &buf[4 * 8]);
	assert(!strncmp(buf, ".......h.......l.......ll.......x", 21));


	return 0;
}

#endif