summaryrefslogtreecommitdiff
path: root/arch/v850/lib/memset.c
blob: d1b2ad821b15a9ec335d90679d5ee35d7927fab0 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
/*
 * arch/v850/lib/memset.c -- Memory initialization
 *
 *  Copyright (C) 2001,02,04  NEC Corporation
 *  Copyright (C) 2001,02,04  Miles Bader <miles@gnu.org>
 *
 * This file is subject to the terms and conditions of the GNU General
 * Public License.  See the file COPYING in the main directory of this
 * archive for more details.
 *
 * Written by Miles Bader <miles@gnu.org>
 */

#include <linux/types.h>

void *memset (void *dst, int val, __kernel_size_t count)
{
	if (count) {
		register unsigned loop;
		register void *ptr asm ("ep") = dst;

		/* replicate VAL into a long.  */
		val &= 0xff;
		val |= val << 8;
		val |= val << 16;

		/* copy initial unaligned bytes.  */
		if ((long)ptr & 1) {
			*(char *)ptr = val;
			ptr = (void *)((char *)ptr + 1);
			count--;
		}
		if (count > 2 && ((long)ptr & 2)) {
			*(short *)ptr = val;
			ptr = (void *)((short *)ptr + 1);
			count -= 2;
		}

		/* 32-byte copying loop.  */
		for (loop = count / 32; loop; loop--) {
			asm ("sst.w %0, 0[ep]; sst.w %0, 4[ep];"
			     "sst.w %0, 8[ep]; sst.w %0, 12[ep];"
			     "sst.w %0, 16[ep]; sst.w %0, 20[ep];"
			     "sst.w %0, 24[ep]; sst.w %0, 28[ep]"
			     :: "r" (val) : "memory");
			ptr += 32;
		}
		count %= 32;

		/* long copying loop.  */
		for (loop = count / 4; loop; loop--) {
			*(long *)ptr = val;
			ptr = (void *)((long *)ptr + 1);
		}
		count %= 4;

		/* finish up with any trailing bytes.  */
		if (count & 2) {
			*(short *)ptr = val;
			ptr = (void *)((short *)ptr + 1);
		}
		if (count & 1) {
			*(char *)ptr = val;
		}
	}

	return dst;
}