xref: /titanic_44/usr/src/lib/libc/sparcv9/gen/memset.s (revision 9a70fc3be3b1e966bf78825cdb8d509963a6f0a1)
17c478bd9Sstevel@tonic-gate/*
27c478bd9Sstevel@tonic-gate * CDDL HEADER START
37c478bd9Sstevel@tonic-gate *
47c478bd9Sstevel@tonic-gate * The contents of this file are subject to the terms of the
58cd45542Sraf * Common Development and Distribution License (the "License").
68cd45542Sraf * You may not use this file except in compliance with the License.
77c478bd9Sstevel@tonic-gate *
87c478bd9Sstevel@tonic-gate * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
97c478bd9Sstevel@tonic-gate * or http://www.opensolaris.org/os/licensing.
107c478bd9Sstevel@tonic-gate * See the License for the specific language governing permissions
117c478bd9Sstevel@tonic-gate * and limitations under the License.
127c478bd9Sstevel@tonic-gate *
137c478bd9Sstevel@tonic-gate * When distributing Covered Code, include this CDDL HEADER in each
147c478bd9Sstevel@tonic-gate * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
157c478bd9Sstevel@tonic-gate * If applicable, add the following below this CDDL HEADER, with the
167c478bd9Sstevel@tonic-gate * fields enclosed by brackets "[]" replaced with your own identifying
177c478bd9Sstevel@tonic-gate * information: Portions Copyright [yyyy] [name of copyright owner]
187c478bd9Sstevel@tonic-gate *
197c478bd9Sstevel@tonic-gate * CDDL HEADER END
207c478bd9Sstevel@tonic-gate */
218cd45542Sraf
227c478bd9Sstevel@tonic-gate/*
238cd45542Sraf * Copyright 2008 Sun Microsystems, Inc.  All rights reserved.
247c478bd9Sstevel@tonic-gate * Use is subject to license terms.
257c478bd9Sstevel@tonic-gate */
267c478bd9Sstevel@tonic-gate
27*9a70fc3bSMark J. Nelson	.file	"memset.s"
287c478bd9Sstevel@tonic-gate
297c478bd9Sstevel@tonic-gate/*
307c478bd9Sstevel@tonic-gate * memset(sp, c, n)
317c478bd9Sstevel@tonic-gate *
327c478bd9Sstevel@tonic-gate * Set an array of n chars starting at sp to the character c.
337c478bd9Sstevel@tonic-gate * Return sp.
347c478bd9Sstevel@tonic-gate *
357c478bd9Sstevel@tonic-gate * Fast assembler language version of the following C-program for memset
367c478bd9Sstevel@tonic-gate * which represents the `standard' for the C-library.
377c478bd9Sstevel@tonic-gate *
387c478bd9Sstevel@tonic-gate *	void *
397c478bd9Sstevel@tonic-gate *	memset(void *sp1, int c, size_t n)
407c478bd9Sstevel@tonic-gate *	{
417c478bd9Sstevel@tonic-gate *	    if (n != 0) {
427c478bd9Sstevel@tonic-gate *		char *sp = sp1;
437c478bd9Sstevel@tonic-gate *		do {
447c478bd9Sstevel@tonic-gate *		    *sp++ = (char)c;
457c478bd9Sstevel@tonic-gate *		} while (--n != 0);
467c478bd9Sstevel@tonic-gate *	    }
477c478bd9Sstevel@tonic-gate *	    return (sp1);
487c478bd9Sstevel@tonic-gate *	}
497c478bd9Sstevel@tonic-gate *
507c478bd9Sstevel@tonic-gate *
517c478bd9Sstevel@tonic-gate *
527c478bd9Sstevel@tonic-gate * Algorithm used:
537c478bd9Sstevel@tonic-gate *	For small stores (6 or fewer bytes), bytes will be stored one at a time.
547c478bd9Sstevel@tonic-gate *
557c478bd9Sstevel@tonic-gate *	When setting 15 or more bytes, there will be at least 8 bytes aligned
567c478bd9Sstevel@tonic-gate *	on an 8-byte boundary.  So, leading bytes will be set, then as many
577c478bd9Sstevel@tonic-gate *	8-byte aligned chunks as possible will be set, followed by any trailing
587c478bd9Sstevel@tonic-gate *	bytes.
597c478bd9Sstevel@tonic-gate *
607c478bd9Sstevel@tonic-gate *	For between 8 and 14 bytes (inclusive), leading odd bytes will be
617c478bd9Sstevel@tonic-gate *	set, followed by 4-byte chunks, followed by trailing bytes.
627c478bd9Sstevel@tonic-gate *
637c478bd9Sstevel@tonic-gate * Inputs:
647c478bd9Sstevel@tonic-gate *	o0:  pointer to start of area to be set to a given value
657c478bd9Sstevel@tonic-gate *	o1:  character used to set memory at location in i0
667c478bd9Sstevel@tonic-gate *	o2:  number of bytes to be set
677c478bd9Sstevel@tonic-gate *
687c478bd9Sstevel@tonic-gate * Outputs:
697c478bd9Sstevel@tonic-gate *	o0:  pointer to start of area set (same as input value in o0)
707c478bd9Sstevel@tonic-gate *
717c478bd9Sstevel@tonic-gate */
727c478bd9Sstevel@tonic-gate
737c478bd9Sstevel@tonic-gate#include <sys/asm_linkage.h>
747c478bd9Sstevel@tonic-gate
757c478bd9Sstevel@tonic-gate	ANSI_PRAGMA_WEAK(memset,function)
767c478bd9Sstevel@tonic-gate
777c478bd9Sstevel@tonic-gate	ENTRY(memset)
787c478bd9Sstevel@tonic-gate	mov	%o0, %o5		! need to return this value
797c478bd9Sstevel@tonic-gate	cmp	%o2, 7
807c478bd9Sstevel@tonic-gate	blu,pn	%xcc, .wrchar		! small count:  just set bytes
817c478bd9Sstevel@tonic-gate	and	%o1, 0xff, %o1
827c478bd9Sstevel@tonic-gate
837c478bd9Sstevel@tonic-gate	sll	%o1, 8, %o4		! generate 4 bytes filled with char
847c478bd9Sstevel@tonic-gate	or 	%o1, %o4, %o1
857c478bd9Sstevel@tonic-gate	sll	%o1, 16, %o4
867c478bd9Sstevel@tonic-gate	cmp	%o2, 15
877c478bd9Sstevel@tonic-gate	blu,pn	%xcc, .walign		! not enough to guarantee 8-byte align
887c478bd9Sstevel@tonic-gate	or	%o1, %o4, %o1
897c478bd9Sstevel@tonic-gate
907c478bd9Sstevel@tonic-gate	sllx	%o1, 32, %o4		! now fill the other 4 bytes with char
917c478bd9Sstevel@tonic-gate	or 	%o1, %o4, %o1
927c478bd9Sstevel@tonic-gate
937c478bd9Sstevel@tonic-gate.dalign:			! Set bytes until 8-byte aligned
947c478bd9Sstevel@tonic-gate	btst	7, %o5			! 8-byte aligned?
957c478bd9Sstevel@tonic-gate	bz,a,pn	%icc, .wrdbl
967c478bd9Sstevel@tonic-gate	andn	%o2, 7, %o3		! o3 has 8-byte multiple
977c478bd9Sstevel@tonic-gate
987c478bd9Sstevel@tonic-gate	dec	%o2
997c478bd9Sstevel@tonic-gate	stb	%o1, [%o5]		! clear a byte
1007c478bd9Sstevel@tonic-gate	b	.dalign			! go see if aligned yet
1017c478bd9Sstevel@tonic-gate	inc	%o5
1027c478bd9Sstevel@tonic-gate
1037c478bd9Sstevel@tonic-gate	.align	32
1047c478bd9Sstevel@tonic-gate.wrdbl:
1057c478bd9Sstevel@tonic-gate	stx	%o1, [%o5]		! write aligned 8 bytes
1067c478bd9Sstevel@tonic-gate	subcc	%o3, 8, %o3
1077c478bd9Sstevel@tonic-gate	bnz,pt	%xcc, .wrdbl
1087c478bd9Sstevel@tonic-gate	inc	8, %o5
1097c478bd9Sstevel@tonic-gate
1107c478bd9Sstevel@tonic-gate	b	.wrchar			! write the remaining bytes
1117c478bd9Sstevel@tonic-gate	and	%o2, 7, %o2		! leftover count, if any
1127c478bd9Sstevel@tonic-gate
1137c478bd9Sstevel@tonic-gate.walign:			! Set bytes until 4-byte aligned
1147c478bd9Sstevel@tonic-gate	btst	3, %o5			! if bigger, align to 4 bytes
1157c478bd9Sstevel@tonic-gate	bz,pn	%icc, .wrword
1167c478bd9Sstevel@tonic-gate	andn	%o2, 3, %o3		! create word sized count in %o3
1177c478bd9Sstevel@tonic-gate
1187c478bd9Sstevel@tonic-gate	dec	%o2			! decrement count
1197c478bd9Sstevel@tonic-gate	stb	%o1, [%o5]		! clear a byte
1207c478bd9Sstevel@tonic-gate	b	.walign
1217c478bd9Sstevel@tonic-gate	inc	%o5			! next byte
1227c478bd9Sstevel@tonic-gate
1237c478bd9Sstevel@tonic-gate.wrword:
1247c478bd9Sstevel@tonic-gate	st	%o1, [%o5]		! 4-byte writing loop
1257c478bd9Sstevel@tonic-gate	subcc	%o3, 4, %o3
1267c478bd9Sstevel@tonic-gate	bnz,pn	%xcc, .wrword
1277c478bd9Sstevel@tonic-gate	inc	4, %o5
1287c478bd9Sstevel@tonic-gate
1297c478bd9Sstevel@tonic-gate	and	%o2, 3, %o2		! leftover count, if any
1307c478bd9Sstevel@tonic-gate
1317c478bd9Sstevel@tonic-gate.wrchar:
1327c478bd9Sstevel@tonic-gate	deccc	%o2			! byte clearing loop
1337c478bd9Sstevel@tonic-gate	inc	%o5
1347c478bd9Sstevel@tonic-gate	bgeu,a,pt %xcc, .wrchar
1357c478bd9Sstevel@tonic-gate	stb	%o1, [%o5 + -1]		! we've already incremented the address
1367c478bd9Sstevel@tonic-gate
1377c478bd9Sstevel@tonic-gate	retl
1387c478bd9Sstevel@tonic-gate	sub	%o0, %g0, %o0
1397c478bd9Sstevel@tonic-gate
1407c478bd9Sstevel@tonic-gate	SET_SIZE(memset)
141