usr/src/lib/libc/i386/gen/strcpy.s
author Mark J. Nelson <Mark.J.Nelson@Sun.COM>
Wed, 06 Aug 2008 16:29:39 -0600
changeset 7298 b69e27387f74
parent 0 68f95e015346
permissions -rw-r--r--
6733918 Teamware has retired, please welcome your new manager, Mercurial 4758439 some files use "current date" sccs keywords 6560843 asm sources should not rely on .file "%M%" for naming STT_FILE symbols 6560958 Solaris:: perl modules should not use SCCS keywords in version information 6729074 webrev doesn't deal well with remote ssh hg parents
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     1
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     2
 * CDDL HEADER START
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     3
 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     4
 * The contents of this file are subject to the terms of the
7298
b69e27387f74 6733918 Teamware has retired, please welcome your new manager, Mercurial
Mark J. Nelson <Mark.J.Nelson@Sun.COM>
parents: 0
diff changeset
     5
 * Common Development and Distribution License (the "License").
b69e27387f74 6733918 Teamware has retired, please welcome your new manager, Mercurial
Mark J. Nelson <Mark.J.Nelson@Sun.COM>
parents: 0
diff changeset
     6
 * You may not use this file except in compliance with the License.
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     7
 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     8
 * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     9
 * or http://www.opensolaris.org/os/licensing.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    10
 * See the License for the specific language governing permissions
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    11
 * and limitations under the License.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    12
 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    13
 * When distributing Covered Code, include this CDDL HEADER in each
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    14
 * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    15
 * If applicable, add the following below this CDDL HEADER, with the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    16
 * fields enclosed by brackets "[]" replaced with your own identifying
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    17
 * information: Portions Copyright [yyyy] [name of copyright owner]
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    18
 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    19
 * CDDL HEADER END
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    20
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    21
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    22
 * Copyright 2005 Sun Microsystems, Inc.  All rights reserved.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    23
 * Use is subject to license terms.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    24
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    25
7298
b69e27387f74 6733918 Teamware has retired, please welcome your new manager, Mercurial
Mark J. Nelson <Mark.J.Nelson@Sun.COM>
parents: 0
diff changeset
    26
	.file	"strcpy.s"
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    27
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    28
/
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    29
/ strcpy(s1, s2)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    30
/
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    31
/ Copies string s2 to s1.  s1 must be large enough.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    32
/ Returns s1
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    33
/
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    34
/
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    35
/ Fast assembly language version of the following C-program strcpy
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    36
/ which represents the `standard' for the C-library.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    37
/
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    38
/	char *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    39
/	strcpy(char *s1, const char *s2)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    40
/	{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    41
/		char	*os1 = s1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    42
/
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    43
/		while (*s1++ = *s2++)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    44
/			;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    45
/		return (os1);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    46
/	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    47
/
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    48
/ In this assembly language version, the following expression is used
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    49
/ to check if a 32-bit word data contains a null byte or not:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    50
/	(((A & 0x7f7f7f7f) + 0x7f7f7f7f) | A) & 0x80808080
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    51
/ If the above expression geneates a value other than 0x80808080,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    52
/ that means the 32-bit word data contains a null byte.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    53
/
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    54
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    55
#include "SYS.h"
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    56
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    57
	ENTRY(strcpy)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    58
	push	%edi				/ save reg as per calling cvntn
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    59
	mov	12(%esp), %ecx			/ src ptr
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    60
	mov	8(%esp), %edi			/ dst ptr
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    61
	mov	%ecx, %eax			/ src
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    62
	sub	%edi, %ecx			/ src - dst
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    63
	and	$3, %eax			/ check src alignment
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    64
	jz	load
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    65
	sub	$4, %eax
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    66
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    67
byte_loop:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    68
	movb	(%edi, %ecx, 1), %dl		/ load src byte
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    69
	movb	%dl, (%edi)			/ load dest byte
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    70
	inc	%edi				/ increment src and dest
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    71
	testb	%dl, %dl			/ is src zero?
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    72
	jz 	done
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    73
	inc	%eax				/ check src alignment
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    74
	jnz	byte_loop
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    75
	jmp 	load
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    76
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    77
store:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    78
	mov	%eax, (%edi)			/ store word
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    79
	add	$4, %edi			/ incrment src and dest by 4
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    80
load:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    81
	mov	(%edi, %ecx, 1), %eax		/ load word
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    82
	lea	-0x01010101(%eax), %edx		/ (word - 0x01010101)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    83
	not	%eax				/ ~word
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    84
	and	%eax, %edx			/ (word - 0x01010101) & ~word
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    85
	not	%eax				/ word
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    86
	and	$0x80808080, %edx	/ (wd - 0x01010101) & ~wd & 0x80808080
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    87
	jz	store				/ store word w/o zero byte
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    88
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    89
has_zero_byte:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    90
	movb	%al, (%edi)			/ store first byte
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    91
	testb	%al, %al			/ check first byte for zero
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    92
	jz	done
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    93
	movb	%ah, 1(%edi)			/ continue storing and checking
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    94
	testb	%ah, %ah
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    95
	jz	done
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    96
	shr	$16, %eax			/ grab last two bytes
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    97
	movb	%al, 2(%edi)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    98
	testb	%al, %al
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    99
	jz	done
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   100
	movb	%ah, 3(%edi)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   101
done:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   102
	mov	8(%esp), %eax			/ return ptr to dest
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   103
	pop	%edi				/ restore as per calling cvntn
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   104
	ret
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   105
	SET_SIZE(strcpy)