1*84d9c625SLionel Sambuc/*- 2*84d9c625SLionel Sambuc * Copyright (c) 2013 The NetBSD Foundation, Inc. 3*84d9c625SLionel Sambuc * All rights reserved. 4*84d9c625SLionel Sambuc * 5*84d9c625SLionel Sambuc * This code is derived from software contributed to The NetBSD Foundation 6*84d9c625SLionel Sambuc * by Matt Thomas of 3am Software Foundry. 7*84d9c625SLionel Sambuc * 8*84d9c625SLionel Sambuc * Redistribution and use in source and binary forms, with or without 9*84d9c625SLionel Sambuc * modification, are permitted provided that the following conditions 10*84d9c625SLionel Sambuc * are met: 11*84d9c625SLionel Sambuc * 1. Redistributions of source code must retain the above copyright 12*84d9c625SLionel Sambuc * notice, this list of conditions and the following disclaimer. 13*84d9c625SLionel Sambuc * 2. Redistributions in binary form must reproduce the above copyright 14*84d9c625SLionel Sambuc * notice, this list of conditions and the following disclaimer in the 15*84d9c625SLionel Sambuc * documentation and/or other materials provided with the distribution. 16*84d9c625SLionel Sambuc * 17*84d9c625SLionel Sambuc * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS 18*84d9c625SLionel Sambuc * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED 19*84d9c625SLionel Sambuc * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR 20*84d9c625SLionel Sambuc * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS 21*84d9c625SLionel Sambuc * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR 22*84d9c625SLionel Sambuc * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF 23*84d9c625SLionel Sambuc * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS 24*84d9c625SLionel Sambuc * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN 25*84d9c625SLionel Sambuc * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) 26*84d9c625SLionel Sambuc * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE 27*84d9c625SLionel Sambuc * POSSIBILITY OF SUCH DAMAGE. 28*84d9c625SLionel Sambuc */ 29*84d9c625SLionel Sambuc#include <machine/asm.h> 30*84d9c625SLionel Sambuc 31*84d9c625SLionel SambucRCSID("$NetBSD: strcpy_thumb.S,v 1.1 2013/08/20 21:32:50 matt Exp $") 32*84d9c625SLionel Sambuc 33*84d9c625SLionel Sambuc#ifdef _LIBC 34*84d9c625SLionel Sambuc#ifdef STRLCPY 35*84d9c625SLionel SambucWEAK_ALIAS(strlcpy, _strlcpy) 36*84d9c625SLionel Sambuc#endif 37*84d9c625SLionel Sambuc#include "namespace.h" 38*84d9c625SLionel Sambuc#endif 39*84d9c625SLionel Sambuc 40*84d9c625SLionel Sambuc/* 41*84d9c625SLionel Sambuc * These are "naive" versions of the str*cpy routines designed to be simple 42*84d9c625SLionel Sambuc * and small. 43*84d9c625SLionel Sambuc */ 44*84d9c625SLionel Sambuc 45*84d9c625SLionel Sambuc#if defined(STRLCPY) 46*84d9c625SLionel Sambuc/* LINTSTUB: size_t strlcpy(char *, const char *, size_t) */ 47*84d9c625SLionel SambucENTRY(strlcpy) 48*84d9c625SLionel Sambuc adds r3, r1, #1 /* save src pointer (+ NUL) */ 49*84d9c625SLionel Sambuc subs r2, r2, #1 /* make sure there's room for a NUL */ 50*84d9c625SLionel Sambuc blt 3f /* no room, do the strlen */ 51*84d9c625SLionel Sambuc adds r2, r2, r0 /* get end of dst */ 52*84d9c625SLionel Sambuc subs r0, r0, r1 /* bias against r1 so one incr */ 53*84d9c625SLionel Sambuc1: cmp r0, r2 /* room for another char? */ 54*84d9c625SLionel Sambuc beq 2f /* no, write NUL and do the strlen */ 55*84d9c625SLionel Sambuc ldrb r3, [r1] /* read a byte */ 56*84d9c625SLionel Sambuc strb r3, [r0, r1] /* write a byte */ 57*84d9c625SLionel Sambuc adds r1, r1, #1 /* advance to next byte */ 58*84d9c625SLionel Sambuc cmp r3, #0 /* was it a NUL? */ 59*84d9c625SLionel Sambuc bne 1b /* no, do next byte */ 60*84d9c625SLionel Sambuc b 4f /* yes, end of string, so return */ 61*84d9c625SLionel Sambuc2: movs r3, #0 /* NUL */ 62*84d9c625SLionel Sambuc strb r3, [r0] /* write to end of string */ 63*84d9c625SLionel Sambuc3: ldrb r3, [r1] /* read a byte */ 64*84d9c625SLionel Sambuc adds r1, r1, #1 /* advance to next byte */ 65*84d9c625SLionel Sambuc cmp r3, #0 /* was it a NUL? */ 66*84d9c625SLionel Sambuc bne 3b /* no, get next byte */ 67*84d9c625SLionel Sambuc4: subs r0, r1, r3 /* return length of src string */ 68*84d9c625SLionel Sambuc RET 69*84d9c625SLionel SambucEND(strlcpy) 70*84d9c625SLionel Sambuc#elif defined(STRNCPY) 71*84d9c625SLionel Sambuc/* LINTSTUB: char * strncpy(char *, const char *, size_t) */ 72*84d9c625SLionel SambucENTRY(strncpy) 73*84d9c625SLionel Sambuc mov ip, r0 /* we want to preserve r0 */ 74*84d9c625SLionel Sambuc adds r2, r2, r0 /* get end of dst buffer */ 75*84d9c625SLionel Sambuc subs r1, r1, r0 /* allows to only increment once */ 76*84d9c625SLionel Sambuc1: cmp r0, r2 /* are at the end of dst already? */ 77*84d9c625SLionel Sambuc beq 3f /* yes, just return. */ 78*84d9c625SLionel Sambuc ldrb r3, [r1, r0] /* read a byte */ 79*84d9c625SLionel Sambuc strb r3, [r0] /* write a byte */ 80*84d9c625SLionel Sambuc adds r0, r0, #1 /* advance a byte */ 81*84d9c625SLionel Sambuc cmp r3, #0 /* was it a NUL? */ 82*84d9c625SLionel Sambuc bne 1b /* no, try next byte */ 83*84d9c625SLionel Sambuc2: cmp r0, r2 /* are at the end of dst already? */ 84*84d9c625SLionel Sambuc beq 3f /* yes, exit loop 85*84d9c625SLionel Sambuc strb r3, [r0] /* write a NUL */ 86*84d9c625SLionel Sambuc adds r0, r0, #1 /* advance a bytes */ 87*84d9c625SLionel Sambuc b 2b /* until dst is filled */ 88*84d9c625SLionel Sambuc3: mov r0, ip /* return dst pointer */ 89*84d9c625SLionel Sambuc RET 90*84d9c625SLionel SambucEND(strncpy) 91*84d9c625SLionel Sambuc#else 92*84d9c625SLionel Sambuc/* LINTSTUB: char * strcpy(char *, const char *) */ 93*84d9c625SLionel SambucENTRY(strcpy) 94*84d9c625SLionel Sambuc subs r2, r0, r1 /* we want to preserve r0 */ 95*84d9c625SLionel Sambuc1: ldrb r3, [r1] /* read a byte */ 96*84d9c625SLionel Sambuc strb r3, [r2, r1] /* write a byte */ 97*84d9c625SLionel Sambuc adds r1, r1, #1 /* advance to next byte */ 98*84d9c625SLionel Sambuc cmp r3, #0 /* was it a NUL? */ 99*84d9c625SLionel Sambuc bne 1b /* no, try next byte */ 100*84d9c625SLionel Sambuc RET /* return dst pointer */ 101*84d9c625SLionel SambucEND(strcpy) 102*84d9c625SLionel Sambuc#endif 103