diff options
Diffstat (limited to 'firmware/lib')
| -rw-r--r-- | firmware/lib/bitops.h | 33 | ||||
| -rw-r--r-- | firmware/lib/changebit.S | 21 | ||||
| -rw-r--r-- | firmware/lib/clearbit.S | 22 | ||||
| -rw-r--r-- | firmware/lib/copy_template.S | 255 | ||||
| -rw-r--r-- | firmware/lib/ctype.c | 34 | ||||
| -rw-r--r-- | firmware/lib/div64.S | 200 | ||||
| -rw-r--r-- | firmware/lib/lib1funcs.S | 338 | ||||
| -rw-r--r-- | firmware/lib/lib_AT91SAM7.c | 405 | ||||
| -rw-r--r-- | firmware/lib/memcpy.S | 59 | ||||
| -rw-r--r-- | firmware/lib/memset.S | 80 | ||||
| -rw-r--r-- | firmware/lib/setbit.S | 22 | ||||
| -rw-r--r-- | firmware/lib/string.c | 41 | ||||
| -rw-r--r-- | firmware/lib/testchangebit.S | 18 | ||||
| -rw-r--r-- | firmware/lib/testclearbit.S | 18 | ||||
| -rw-r--r-- | firmware/lib/testsetbit.S | 18 | ||||
| -rw-r--r-- | firmware/lib/vsprintf.c | 830 | 
16 files changed, 2394 insertions, 0 deletions
| diff --git a/firmware/lib/bitops.h b/firmware/lib/bitops.h new file mode 100644 index 0000000..428c9a6 --- /dev/null +++ b/firmware/lib/bitops.h @@ -0,0 +1,33 @@ +	.macro	bitop, instr +	and	r2, r0, #7 +	mov	r3, #1 +	mov	r3, r3, lsl r2 +	save_and_disable_irqs ip +	ldrb	r2, [r1, r0, lsr #3] +	\instr	r2, r2, r3 +	strb	r2, [r1, r0, lsr #3] +	restore_irqs ip +	mov	pc, lr +	.endm + +/** + * testop - implement a test_and_xxx_bit operation. + * @instr: operational instruction + * @store: store instruction + * + * Note: we can trivially conditionalise the store instruction + * to avoid dirting the data cache. + */ +	.macro	testop, instr, store +	add	r1, r1, r0, lsr #3 +	and	r3, r0, #7 +	mov	r0, #1 +	save_and_disable_irqs ip +	ldrb	r2, [r1] +	tst	r2, r0, lsl r3 +	\instr	r2, r2, r0, lsl r3 +	\store	r2, [r1] +	restore_irqs ip +	moveq	r0, #0 +	mov	pc, lr +	.endm diff --git a/firmware/lib/changebit.S b/firmware/lib/changebit.S new file mode 100644 index 0000000..7c709fb --- /dev/null +++ b/firmware/lib/changebit.S @@ -0,0 +1,21 @@ +/* + *  linux/arch/arm/lib/changebit.S + * + *  Copyright (C) 1995-1996 Russell King + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License version 2 as + * published by the Free Software Foundation. + */ +#include <asm/linkage.h> +#include <asm/assembler.h> +#include "bitops.h" +                .text + +/* Purpose  : Function to change a bit + * Prototype: int change_bit(int bit, void *addr) + */ +ENTRY(_change_bit_be) +		eor	r0, r0, #0x18		@ big endian byte ordering +ENTRY(_change_bit_le) +	bitop	eor diff --git a/firmware/lib/clearbit.S b/firmware/lib/clearbit.S new file mode 100644 index 0000000..cb48f7a --- /dev/null +++ b/firmware/lib/clearbit.S @@ -0,0 +1,22 @@ +/* + *  linux/arch/arm/lib/clearbit.S + * + *  Copyright (C) 1995-1996 Russell King + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License version 2 as + * published by the Free Software Foundation. + */ +#include <asm/linkage.h> +#include <asm/assembler.h> +#include "bitops.h" +                .text + +/* + * Purpose  : Function to clear a bit + * Prototype: int clear_bit(int bit, void *addr) + */ +ENTRY(_clear_bit_be) +		eor	r0, r0, #0x18		@ big endian byte ordering +ENTRY(_clear_bit_le) +	bitop	bic diff --git a/firmware/lib/copy_template.S b/firmware/lib/copy_template.S new file mode 100644 index 0000000..cab355c --- /dev/null +++ b/firmware/lib/copy_template.S @@ -0,0 +1,255 @@ +/* + *  linux/arch/arm/lib/copy_template.s + * + *  Code template for optimized memory copy functions + * + *  Author:	Nicolas Pitre + *  Created:	Sep 28, 2005 + *  Copyright:	MontaVista Software, Inc. + * + *  This program is free software; you can redistribute it and/or modify + *  it under the terms of the GNU General Public License version 2 as + *  published by the Free Software Foundation. + */ + +/* + * This can be used to enable code to cacheline align the source pointer. + * Experiments on tested architectures (StrongARM and XScale) didn't show + * this a worthwhile thing to do.  That might be different in the future. + */ +//#define CALGN(code...)	code +#define CALGN(code...) + +/* + * Theory of operation + * ------------------- + * + * This file provides the core code for a forward memory copy used in + * the implementation of memcopy(), copy_to_user() and copy_from_user(). + * + * The including file must define the following accessor macros + * according to the need of the given function: + * + * ldr1w ptr reg abort + * + *	This loads one word from 'ptr', stores it in 'reg' and increments + *	'ptr' to the next word. The 'abort' argument is used for fixup tables. + * + * ldr4w ptr reg1 reg2 reg3 reg4 abort + * ldr8w ptr, reg1 reg2 reg3 reg4 reg5 reg6 reg7 reg8 abort + * + *	This loads four or eight words starting from 'ptr', stores them + *	in provided registers and increments 'ptr' past those words. + *	The'abort' argument is used for fixup tables. + * + * ldr1b ptr reg cond abort + * + *	Similar to ldr1w, but it loads a byte and increments 'ptr' one byte. + *	It also must apply the condition code if provided, otherwise the + *	"al" condition is assumed by default. + * + * str1w ptr reg abort + * str8w ptr reg1 reg2 reg3 reg4 reg5 reg6 reg7 reg8 abort + * str1b ptr reg cond abort + * + *	Same as their ldr* counterparts, but data is stored to 'ptr' location + *	rather than being loaded. + * + * enter reg1 reg2 + * + *	Preserve the provided registers on the stack plus any additional + *	data as needed by the implementation including this code. Called + *	upon code entry. + * + * exit reg1 reg2 + * + *	Restore registers with the values previously saved with the + *	'preserv' macro. Called upon code termination. + */ + + +		enter	r4, lr + +		subs	r2, r2, #4 +		blt	8f +		ands	ip, r0, #3 +	PLD(	pld	[r1, #0]		) +		bne	9f +		ands	ip, r1, #3 +		bne	10f + +1:		subs	r2, r2, #(28) +		stmfd	sp!, {r5 - r8} +		blt	5f + +	CALGN(	ands	ip, r1, #31		) +	CALGN(	rsb	r3, ip, #32		) +	CALGN(	sbcnes	r4, r3, r2		)  @ C is always set here +	CALGN(	bcs	2f			) +	CALGN(	adr	r4, 6f			) +	CALGN(	subs	r2, r2, r3		)  @ C gets set +	CALGN(	add	pc, r4, ip		) + +	PLD(	pld	[r1, #0]		) +2:	PLD(	subs	r2, r2, #96		) +	PLD(	pld	[r1, #28]		) +	PLD(	blt	4f			) +	PLD(	pld	[r1, #60]		) +	PLD(	pld	[r1, #92]		) + +3:	PLD(	pld	[r1, #124]		) +4:		ldr8w	r1, r3, r4, r5, r6, r7, r8, ip, lr, abort=20f +		subs	r2, r2, #32 +		str8w	r0, r3, r4, r5, r6, r7, r8, ip, lr, abort=20f +		bge	3b +	PLD(	cmn	r2, #96			) +	PLD(	bge	4b			) + +5:		ands	ip, r2, #28 +		rsb	ip, ip, #32 +		addne	pc, pc, ip		@ C is always clear here +		b	7f +6:		nop +		ldr1w	r1, r3, abort=20f +		ldr1w	r1, r4, abort=20f +		ldr1w	r1, r5, abort=20f +		ldr1w	r1, r6, abort=20f +		ldr1w	r1, r7, abort=20f +		ldr1w	r1, r8, abort=20f +		ldr1w	r1, lr, abort=20f + +		add	pc, pc, ip +		nop +		nop +		str1w	r0, r3, abort=20f +		str1w	r0, r4, abort=20f +		str1w	r0, r5, abort=20f +		str1w	r0, r6, abort=20f +		str1w	r0, r7, abort=20f +		str1w	r0, r8, abort=20f +		str1w	r0, lr, abort=20f + +	CALGN(	bcs	2b			) + +7:		ldmfd	sp!, {r5 - r8} + +8:		movs	r2, r2, lsl #31 +		ldr1b	r1, r3, ne, abort=21f +		ldr1b	r1, r4, cs, abort=21f +		ldr1b	r1, ip, cs, abort=21f +		str1b	r0, r3, ne, abort=21f +		str1b	r0, r4, cs, abort=21f +		str1b	r0, ip, cs, abort=21f + +		exit	r4, pc + +9:		rsb	ip, ip, #4 +		cmp	ip, #2 +		ldr1b	r1, r3, gt, abort=21f +		ldr1b	r1, r4, ge, abort=21f +		ldr1b	r1, lr, abort=21f +		str1b	r0, r3, gt, abort=21f +		str1b	r0, r4, ge, abort=21f +		subs	r2, r2, ip +		str1b	r0, lr, abort=21f +		blt	8b +		ands	ip, r1, #3 +		beq	1b + +10:		bic	r1, r1, #3 +		cmp	ip, #2 +		ldr1w	r1, lr, abort=21f +		beq	17f +		bgt	18f + + +		.macro	forward_copy_shift pull push + +		subs	r2, r2, #28 +		blt	14f + +	CALGN(	ands	ip, r1, #31		) +	CALGN(	rsb	ip, ip, #32		) +	CALGN(	sbcnes	r4, ip, r2		)  @ C is always set here +	CALGN(	subcc	r2, r2, ip		) +	CALGN(	bcc	15f			) + +11:		stmfd	sp!, {r5 - r9} + +	PLD(	pld	[r1, #0]		) +	PLD(	subs	r2, r2, #96		) +	PLD(	pld	[r1, #28]		) +	PLD(	blt	13f			) +	PLD(	pld	[r1, #60]		) +	PLD(	pld	[r1, #92]		) + +12:	PLD(	pld	[r1, #124]		) +13:		ldr4w	r1, r4, r5, r6, r7, abort=19f +		mov	r3, lr, pull #\pull +		subs	r2, r2, #32 +		ldr4w	r1, r8, r9, ip, lr, abort=19f +		orr	r3, r3, r4, push #\push +		mov	r4, r4, pull #\pull +		orr	r4, r4, r5, push #\push +		mov	r5, r5, pull #\pull +		orr	r5, r5, r6, push #\push +		mov	r6, r6, pull #\pull +		orr	r6, r6, r7, push #\push +		mov	r7, r7, pull #\pull +		orr	r7, r7, r8, push #\push +		mov	r8, r8, pull #\pull +		orr	r8, r8, r9, push #\push +		mov	r9, r9, pull #\pull +		orr	r9, r9, ip, push #\push +		mov	ip, ip, pull #\pull +		orr	ip, ip, lr, push #\push +		str8w	r0, r3, r4, r5, r6, r7, r8, r9, ip, , abort=19f +		bge	12b +	PLD(	cmn	r2, #96			) +	PLD(	bge	13b			) + +		ldmfd	sp!, {r5 - r9} + +14:		ands	ip, r2, #28 +		beq	16f + +15:		mov	r3, lr, pull #\pull +		ldr1w	r1, lr, abort=21f +		subs	ip, ip, #4 +		orr	r3, r3, lr, push #\push +		str1w	r0, r3, abort=21f +		bgt	15b +	CALGN(	cmp	r2, #0			) +	CALGN(	bge	11b			) + +16:		sub	r1, r1, #(\push / 8) +		b	8b + +		.endm + + +		forward_copy_shift	pull=8	push=24 + +17:		forward_copy_shift	pull=16	push=16 + +18:		forward_copy_shift	pull=24	push=8 + + +/* + * Abort preamble and completion macros. + * If a fixup handler is required then those macros must surround it. + * It is assumed that the fixup code will handle the private part of + * the exit macro. + */ + +	.macro	copy_abort_preamble +19:	ldmfd	sp!, {r5 - r9} +	b	21f +20:	ldmfd	sp!, {r5 - r8} +21: +	.endm + +	.macro	copy_abort_end +	ldmfd	sp!, {r4, pc} +	.endm + diff --git a/firmware/lib/ctype.c b/firmware/lib/ctype.c new file mode 100644 index 0000000..6ec51cc --- /dev/null +++ b/firmware/lib/ctype.c @@ -0,0 +1,34 @@ +/* + *  linux/lib/ctype.c + * + *  Copyright (C) 1991, 1992  Linus Torvalds + */ + +#include <asm/ctype.h> + +unsigned char _ctype[] = { +_C,_C,_C,_C,_C,_C,_C,_C,			/* 0-7 */ +_C,_C|_S,_C|_S,_C|_S,_C|_S,_C|_S,_C,_C,		/* 8-15 */ +_C,_C,_C,_C,_C,_C,_C,_C,			/* 16-23 */ +_C,_C,_C,_C,_C,_C,_C,_C,			/* 24-31 */ +_S|_SP,_P,_P,_P,_P,_P,_P,_P,			/* 32-39 */ +_P,_P,_P,_P,_P,_P,_P,_P,			/* 40-47 */ +_D,_D,_D,_D,_D,_D,_D,_D,			/* 48-55 */ +_D,_D,_P,_P,_P,_P,_P,_P,			/* 56-63 */ +_P,_U|_X,_U|_X,_U|_X,_U|_X,_U|_X,_U|_X,_U,	/* 64-71 */ +_U,_U,_U,_U,_U,_U,_U,_U,			/* 72-79 */ +_U,_U,_U,_U,_U,_U,_U,_U,			/* 80-87 */ +_U,_U,_U,_P,_P,_P,_P,_P,			/* 88-95 */ +_P,_L|_X,_L|_X,_L|_X,_L|_X,_L|_X,_L|_X,_L,	/* 96-103 */ +_L,_L,_L,_L,_L,_L,_L,_L,			/* 104-111 */ +_L,_L,_L,_L,_L,_L,_L,_L,			/* 112-119 */ +_L,_L,_L,_P,_P,_P,_P,_C,			/* 120-127 */ +0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,		/* 128-143 */ +0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,		/* 144-159 */ +_S|_SP,_P,_P,_P,_P,_P,_P,_P,_P,_P,_P,_P,_P,_P,_P,_P,   /* 160-175 */ +_P,_P,_P,_P,_P,_P,_P,_P,_P,_P,_P,_P,_P,_P,_P,_P,       /* 176-191 */ +_U,_U,_U,_U,_U,_U,_U,_U,_U,_U,_U,_U,_U,_U,_U,_U,       /* 192-207 */ +_U,_U,_U,_U,_U,_U,_U,_P,_U,_U,_U,_U,_U,_U,_U,_L,       /* 208-223 */ +_L,_L,_L,_L,_L,_L,_L,_L,_L,_L,_L,_L,_L,_L,_L,_L,       /* 224-239 */ +_L,_L,_L,_L,_L,_L,_L,_P,_L,_L,_L,_L,_L,_L,_L,_L};      /* 240-255 */ + diff --git a/firmware/lib/div64.S b/firmware/lib/div64.S new file mode 100644 index 0000000..7eeef50 --- /dev/null +++ b/firmware/lib/div64.S @@ -0,0 +1,200 @@ +/* + *  linux/arch/arm/lib/div64.S + * + *  Optimized computation of 64-bit dividend / 32-bit divisor + * + *  Author:	Nicolas Pitre + *  Created:	Oct 5, 2003 + *  Copyright:	Monta Vista Software, Inc. + * + *  This program is free software; you can redistribute it and/or modify + *  it under the terms of the GNU General Public License version 2 as + *  published by the Free Software Foundation. + */ + +#include <asm/linkage.h> + +#ifdef __ARMEB__ +#define xh r0 +#define xl r1 +#define yh r2 +#define yl r3 +#else +#define xl r0 +#define xh r1 +#define yl r2 +#define yh r3 +#endif + +/* + * __do_div64: perform a division with 64-bit dividend and 32-bit divisor. + * + * Note: Calling convention is totally non standard for optimal code. + *       This is meant to be used by do_div() from include/asm/div64.h only. + * + * Input parameters: + * 	xh-xl	= dividend (clobbered) + * 	r4	= divisor (preserved) + * + * Output values: + * 	yh-yl	= result + * 	xh	= remainder + * + * Clobbered regs: xl, ip + */ + +ENTRY(__do_div64) + +	@ Test for easy paths first. +	subs	ip, r4, #1 +	bls	9f			@ divisor is 0 or 1 +	tst	ip, r4 +	beq	8f			@ divisor is power of 2 + +	@ See if we need to handle upper 32-bit result. +	cmp	xh, r4 +	mov	yh, #0 +	blo	3f + +	@ Align divisor with upper part of dividend. +	@ The aligned divisor is stored in yl preserving the original. +	@ The bit position is stored in ip. + +#if __LINUX_ARM_ARCH__ >= 5 + +	clz	yl, r4 +	clz	ip, xh +	sub	yl, yl, ip +	mov	ip, #1 +	mov	ip, ip, lsl yl +	mov	yl, r4, lsl yl + +#else + +	mov	yl, r4 +	mov	ip, #1 +1:	cmp	yl, #0x80000000 +	cmpcc	yl, xh +	movcc	yl, yl, lsl #1 +	movcc	ip, ip, lsl #1 +	bcc	1b + +#endif + +	@ The division loop for needed upper bit positions. + 	@ Break out early if dividend reaches 0. +2:	cmp	xh, yl +	orrcs	yh, yh, ip +	subcss	xh, xh, yl +	movnes	ip, ip, lsr #1 +	mov	yl, yl, lsr #1 +	bne	2b + +	@ See if we need to handle lower 32-bit result. +3:	cmp	xh, #0 +	mov	yl, #0 +	cmpeq	xl, r4 +	movlo	xh, xl +	movlo	pc, lr + +	@ The division loop for lower bit positions. +	@ Here we shift remainer bits leftwards rather than moving the +	@ divisor for comparisons, considering the carry-out bit as well. +	mov	ip, #0x80000000 +4:	movs	xl, xl, lsl #1 +	adcs	xh, xh, xh +	beq	6f +	cmpcc	xh, r4 +5:	orrcs	yl, yl, ip +	subcs	xh, xh, r4 +	movs	ip, ip, lsr #1 +	bne	4b +	mov	pc, lr + +	@ The top part of remainder became zero.  If carry is set +	@ (the 33th bit) this is a false positive so resume the loop. +	@ Otherwise, if lower part is also null then we are done. +6:	bcs	5b +	cmp	xl, #0 +	moveq	pc, lr + +	@ We still have remainer bits in the low part.  Bring them up. + +#if __LINUX_ARM_ARCH__ >= 5 + +	clz	xh, xl			@ we know xh is zero here so... +	add	xh, xh, #1 +	mov	xl, xl, lsl xh +	mov	ip, ip, lsr xh + +#else + +7:	movs	xl, xl, lsl #1 +	mov	ip, ip, lsr #1 +	bcc	7b + +#endif + +	@ Current remainder is now 1.  It is worthless to compare with +	@ divisor at this point since divisor can not be smaller than 3 here. +	@ If possible, branch for another shift in the division loop. +	@ If no bit position left then we are done. +	movs	ip, ip, lsr #1 +	mov	xh, #1 +	bne	4b +	mov	pc, lr + +8:	@ Division by a power of 2: determine what that divisor order is +	@ then simply shift values around + +#if __LINUX_ARM_ARCH__ >= 5 + +	clz	ip, r4 +	rsb	ip, ip, #31 + +#else + +	mov	yl, r4 +	cmp	r4, #(1 << 16) +	mov	ip, #0 +	movhs	yl, yl, lsr #16 +	movhs	ip, #16 + +	cmp	yl, #(1 << 8) +	movhs	yl, yl, lsr #8 +	addhs	ip, ip, #8 + +	cmp	yl, #(1 << 4) +	movhs	yl, yl, lsr #4 +	addhs	ip, ip, #4 + +	cmp	yl, #(1 << 2) +	addhi	ip, ip, #3 +	addls	ip, ip, yl, lsr #1 + +#endif + +	mov	yh, xh, lsr ip +	mov	yl, xl, lsr ip +	rsb	ip, ip, #32 +	orr	yl, yl, xh, lsl ip +	mov	xh, xl, lsl ip +	mov	xh, xh, lsr ip +	mov	pc, lr + +	@ eq -> division by 1: obvious enough... +9:	moveq	yl, xl +	moveq	yh, xh +	moveq	xh, #0 +	moveq	pc, lr + +	@ Division by 0: +	str	lr, [sp, #-8]! +	bl	__div0 + +	@ as wrong as it could be... +	mov	yl, #0 +	mov	yh, #0 +	mov	xh, #0 +	ldr	pc, [sp], #8 + diff --git a/firmware/lib/lib1funcs.S b/firmware/lib/lib1funcs.S new file mode 100644 index 0000000..92a6c56 --- /dev/null +++ b/firmware/lib/lib1funcs.S @@ -0,0 +1,338 @@ +/* + * linux/arch/arm/lib/lib1funcs.S: Optimized ARM division routines + * + * Author: Nicolas Pitre <nico@cam.org> + *   - contributed to gcc-3.4 on Sep 30, 2003 + *   - adapted for the Linux kernel on Oct 2, 2003 + */ + +/* Copyright 1995, 1996, 1998, 1999, 2000, 2003 Free Software Foundation, Inc. + +This file is free software; you can redistribute it and/or modify it +under the terms of the GNU General Public License as published by the +Free Software Foundation; either version 2, or (at your option) any +later version. + +In addition to the permissions in the GNU General Public License, the +Free Software Foundation gives you unlimited permission to link the +compiled version of this file into combinations with other programs, +and to distribute those combinations without any restriction coming +from the use of this file.  (The General Public License restrictions +do apply in other respects; for example, they cover modification of +the file, and distribution when not linked into a combine +executable.) + +This file is distributed in the hope that it will be useful, but +WITHOUT ANY WARRANTY; without even the implied warranty of +MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU +General Public License for more details. + +You should have received a copy of the GNU General Public License +along with this program; see the file COPYING.  If not, write to +the Free Software Foundation, 59 Temple Place - Suite 330, +Boston, MA 02111-1307, USA.  */ + + +#include <asm/linkage.h> +#include <asm/assembler.h> + + +.macro ARM_DIV_BODY dividend, divisor, result, curbit + +#if __LINUX_ARM_ARCH__ >= 5 + +	clz	\curbit, \divisor +	clz	\result, \dividend +	sub	\result, \curbit, \result +	mov	\curbit, #1 +	mov	\divisor, \divisor, lsl \result +	mov	\curbit, \curbit, lsl \result +	mov	\result, #0 +	 +#else + +	@ Initially shift the divisor left 3 bits if possible, +	@ set curbit accordingly.  This allows for curbit to be located +	@ at the left end of each 4 bit nibbles in the division loop +	@ to save one loop in most cases. +	tst	\divisor, #0xe0000000 +	moveq	\divisor, \divisor, lsl #3 +	moveq	\curbit, #8 +	movne	\curbit, #1 + +	@ Unless the divisor is very big, shift it up in multiples of +	@ four bits, since this is the amount of unwinding in the main +	@ division loop.  Continue shifting until the divisor is  +	@ larger than the dividend. +1:	cmp	\divisor, #0x10000000 +	cmplo	\divisor, \dividend +	movlo	\divisor, \divisor, lsl #4 +	movlo	\curbit, \curbit, lsl #4 +	blo	1b + +	@ For very big divisors, we must shift it a bit at a time, or +	@ we will be in danger of overflowing. +1:	cmp	\divisor, #0x80000000 +	cmplo	\divisor, \dividend +	movlo	\divisor, \divisor, lsl #1 +	movlo	\curbit, \curbit, lsl #1 +	blo	1b + +	mov	\result, #0 + +#endif + +	@ Division loop +1:	cmp	\dividend, \divisor +	subhs	\dividend, \dividend, \divisor +	orrhs	\result,   \result,   \curbit +	cmp	\dividend, \divisor,  lsr #1 +	subhs	\dividend, \dividend, \divisor, lsr #1 +	orrhs	\result,   \result,   \curbit,  lsr #1 +	cmp	\dividend, \divisor,  lsr #2 +	subhs	\dividend, \dividend, \divisor, lsr #2 +	orrhs	\result,   \result,   \curbit,  lsr #2 +	cmp	\dividend, \divisor,  lsr #3 +	subhs	\dividend, \dividend, \divisor, lsr #3 +	orrhs	\result,   \result,   \curbit,  lsr #3 +	cmp	\dividend, #0			@ Early termination? +	movnes	\curbit,   \curbit,  lsr #4	@ No, any more bits to do? +	movne	\divisor,  \divisor, lsr #4 +	bne	1b + +.endm + + +.macro ARM_DIV2_ORDER divisor, order + +#if __LINUX_ARM_ARCH__ >= 5 + +	clz	\order, \divisor +	rsb	\order, \order, #31 + +#else + +	cmp	\divisor, #(1 << 16) +	movhs	\divisor, \divisor, lsr #16 +	movhs	\order, #16 +	movlo	\order, #0 + +	cmp	\divisor, #(1 << 8) +	movhs	\divisor, \divisor, lsr #8 +	addhs	\order, \order, #8 + +	cmp	\divisor, #(1 << 4) +	movhs	\divisor, \divisor, lsr #4 +	addhs	\order, \order, #4 + +	cmp	\divisor, #(1 << 2) +	addhi	\order, \order, #3 +	addls	\order, \order, \divisor, lsr #1 + +#endif + +.endm + + +.macro ARM_MOD_BODY dividend, divisor, order, spare + +#if __LINUX_ARM_ARCH__ >= 5 + +	clz	\order, \divisor +	clz	\spare, \dividend +	sub	\order, \order, \spare +	mov	\divisor, \divisor, lsl \order + +#else + +	mov	\order, #0 + +	@ Unless the divisor is very big, shift it up in multiples of +	@ four bits, since this is the amount of unwinding in the main +	@ division loop.  Continue shifting until the divisor is  +	@ larger than the dividend. +1:	cmp	\divisor, #0x10000000 +	cmplo	\divisor, \dividend +	movlo	\divisor, \divisor, lsl #4 +	addlo	\order, \order, #4 +	blo	1b + +	@ For very big divisors, we must shift it a bit at a time, or +	@ we will be in danger of overflowing. +1:	cmp	\divisor, #0x80000000 +	cmplo	\divisor, \dividend +	movlo	\divisor, \divisor, lsl #1 +	addlo	\order, \order, #1 +	blo	1b + +#endif + +	@ Perform all needed substractions to keep only the reminder. +	@ Do comparisons in batch of 4 first. +	subs	\order, \order, #3		@ yes, 3 is intended here +	blt	2f + +1:	cmp	\dividend, \divisor +	subhs	\dividend, \dividend, \divisor +	cmp	\dividend, \divisor,  lsr #1 +	subhs	\dividend, \dividend, \divisor, lsr #1 +	cmp	\dividend, \divisor,  lsr #2 +	subhs	\dividend, \dividend, \divisor, lsr #2 +	cmp	\dividend, \divisor,  lsr #3 +	subhs	\dividend, \dividend, \divisor, lsr #3 +	cmp	\dividend, #1 +	mov	\divisor, \divisor, lsr #4 +	subges	\order, \order, #4 +	bge	1b + +	tst	\order, #3 +	teqne	\dividend, #0 +	beq	5f + +	@ Either 1, 2 or 3 comparison/substractions are left. +2:	cmn	\order, #2 +	blt	4f +	beq	3f +	cmp	\dividend, \divisor +	subhs	\dividend, \dividend, \divisor +	mov	\divisor,  \divisor,  lsr #1 +3:	cmp	\dividend, \divisor +	subhs	\dividend, \dividend, \divisor +	mov	\divisor,  \divisor,  lsr #1 +4:	cmp	\dividend, \divisor +	subhs	\dividend, \dividend, \divisor +5: +.endm + + +ENTRY(__udivsi3) +ENTRY(__aeabi_uidiv) + +	subs	r2, r1, #1 +	moveq	pc, lr +	bcc	Ldiv0 +	cmp	r0, r1 +	bls	11f +	tst	r1, r2 +	beq	12f + +	ARM_DIV_BODY r0, r1, r2, r3 + +	mov	r0, r2 +	mov	pc, lr + +11:	moveq	r0, #1 +	movne	r0, #0 +	mov	pc, lr + +12:	ARM_DIV2_ORDER r1, r2 + +	mov	r0, r0, lsr r2 +	mov	pc, lr + + +ENTRY(__umodsi3) + +	subs	r2, r1, #1			@ compare divisor with 1 +	bcc	Ldiv0 +	cmpne	r0, r1				@ compare dividend with divisor +	moveq   r0, #0 +	tsthi	r1, r2				@ see if divisor is power of 2 +	andeq	r0, r0, r2 +	movls	pc, lr + +	ARM_MOD_BODY r0, r1, r2, r3 + +	mov	pc, lr + + +ENTRY(__divsi3) +ENTRY(__aeabi_idiv) + +	cmp	r1, #0 +	eor	ip, r0, r1			@ save the sign of the result. +	beq	Ldiv0 +	rsbmi	r1, r1, #0			@ loops below use unsigned. +	subs	r2, r1, #1			@ division by 1 or -1 ? +	beq	10f +	movs	r3, r0 +	rsbmi	r3, r0, #0			@ positive dividend value +	cmp	r3, r1 +	bls	11f +	tst	r1, r2				@ divisor is power of 2 ? +	beq	12f + +	ARM_DIV_BODY r3, r1, r0, r2 + +	cmp	ip, #0 +	rsbmi	r0, r0, #0 +	mov	pc, lr + +10:	teq	ip, r0				@ same sign ? +	rsbmi	r0, r0, #0 +	mov	pc, lr + +11:	movlo	r0, #0 +	moveq	r0, ip, asr #31 +	orreq	r0, r0, #1 +	mov	pc, lr + +12:	ARM_DIV2_ORDER r1, r2 + +	cmp	ip, #0 +	mov	r0, r3, lsr r2 +	rsbmi	r0, r0, #0 +	mov	pc, lr + + +ENTRY(__modsi3) + +	cmp	r1, #0 +	beq	Ldiv0 +	rsbmi	r1, r1, #0			@ loops below use unsigned. +	movs	ip, r0				@ preserve sign of dividend +	rsbmi	r0, r0, #0			@ if negative make positive +	subs	r2, r1, #1			@ compare divisor with 1 +	cmpne	r0, r1				@ compare dividend with divisor +	moveq	r0, #0 +	tsthi	r1, r2				@ see if divisor is power of 2 +	andeq	r0, r0, r2 +	bls	10f + +	ARM_MOD_BODY r0, r1, r2, r3 + +10:	cmp	ip, #0 +	rsbmi	r0, r0, #0 +	mov	pc, lr + +#ifdef CONFIG_AEABI + +ENTRY(__aeabi_uidivmod) + +	stmfd	sp!, {r0, r1, ip, lr} +	bl	__aeabi_uidiv +	ldmfd	sp!, {r1, r2, ip, lr} +	mul	r3, r0, r2 +	sub	r1, r1, r3 +	mov	pc, lr + +ENTRY(__aeabi_idivmod) + +	stmfd	sp!, {r0, r1, ip, lr} +	bl	__aeabi_idiv +	ldmfd	sp!, {r1, r2, ip, lr} +	mul	r3, r0, r2 +	sub	r1, r1, r3 +	mov	pc, lr + +#endif + +Ldiv0: + +	str	lr, [sp, #-8]! +	bl	__div0 +	mov	r0, #0			@ About as wrong as it could be. +	ldr	pc, [sp], #8 + +ENTRY(__div0) +	mov	pc, lr diff --git a/firmware/lib/lib_AT91SAM7.c b/firmware/lib/lib_AT91SAM7.c new file mode 100644 index 0000000..950761f --- /dev/null +++ b/firmware/lib/lib_AT91SAM7.c @@ -0,0 +1,405 @@ +//* ---------------------------------------------------------------------------- +//*         ATMEL Microcontroller Software Support  -  ROUSSET  - +//* ---------------------------------------------------------------------------- +//* DISCLAIMER:  THIS SOFTWARE IS PROVIDED BY ATMEL "AS IS" AND ANY EXPRESS OR +//* IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF +//* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT ARE +//* DISCLAIMED. IN NO EVENT SHALL ATMEL BE LIABLE FOR ANY DIRECT, INDIRECT, +//* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +//* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, +//* OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF +//* LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING +//* NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, +//* EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. +//* ---------------------------------------------------------------------------- +//* File Name           : lib_AT91SAM7S64.h +//* Object              : AT91SAM7S64 inlined functions +//* Generated           : AT91 SW Application Group  08/30/2005 (15:52:59) +//* + +#include <sys/types.h> +#include <include/AT91SAM7.h> +#include <include/lib_AT91SAM7.h> + +//*---------------------------------------------------------------------------- +//* \fn    AT91F_AIC_ConfigureIt +//* \brief Interrupt Handler Initialization +//*---------------------------------------------------------------------------- +unsigned int AT91F_AIC_ConfigureIt ( +	AT91PS_AIC pAic,  // \arg pointer to the AIC registers +	unsigned int irq_id,     // \arg interrupt number to initialize +	unsigned int priority,   // \arg priority to give to the interrupt +	unsigned int src_type,   // \arg activation and sense of activation +	void (*newHandler) () ) // \arg address of the interrupt handler +{ +	unsigned int oldHandler; +    unsigned int mask ; + +    oldHandler = pAic->AIC_SVR[irq_id]; + +    mask = 0x1 << irq_id ; +    //* Disable the interrupt on the interrupt controller +    pAic->AIC_IDCR = mask ; +    //* Save the interrupt handler routine pointer and the interrupt priority +    pAic->AIC_SVR[irq_id] = (unsigned int) newHandler ; +    //* Store the Source Mode Register +    pAic->AIC_SMR[irq_id] = src_type | priority  ; +    //* Clear the interrupt on the interrupt controller +    pAic->AIC_ICCR = mask ; + +	return oldHandler; +} + +//*---------------------------------------------------------------------------- +//* \fn    AT91F_AIC_SetExceptionVector +//* \brief Configure vector handler +//*---------------------------------------------------------------------------- +unsigned int  AT91F_AIC_SetExceptionVector ( +	unsigned int *pVector, // \arg pointer to the AIC registers +	void (*Handler) () )   // \arg Interrupt Handler +{ +	unsigned int oldVector = *pVector; + +	if ((unsigned int) Handler == (unsigned int) AT91C_AIC_BRANCH_OPCODE) +		*pVector = (unsigned int) AT91C_AIC_BRANCH_OPCODE; +	else +		*pVector = (((((unsigned int) Handler) - ((unsigned int) pVector) - 0x8) >> 2) & 0x00FFFFFF) | 0xEA000000; + +	return oldVector; +} + +//*---------------------------------------------------------------------------- +//* \fn    AT91F_AIC_Open +//* \brief Set exception vectors and AIC registers to default values +//*---------------------------------------------------------------------------- +void AT91F_AIC_Open( +	AT91PS_AIC pAic,        // \arg pointer to the AIC registers +	void (*IrqHandler) (),  // \arg Default IRQ vector exception +	void (*FiqHandler) (),  // \arg Default FIQ vector exception +	void (*DefaultHandler)  (), // \arg Default Handler set in ISR +	void (*SpuriousHandler) (), // \arg Default Spurious Handler +	unsigned int protectMode)   // \arg Debug Control Register +{ +	int i; + +	// Disable all interrupts and set IVR to the default handler +	for (i = 0; i < 32; ++i) { +		AT91F_AIC_DisableIt(pAic, i); +		AT91F_AIC_ConfigureIt(pAic, i, AT91C_AIC_PRIOR_LOWEST, AT91C_AIC_SRCTYPE_HIGH_LEVEL, DefaultHandler); +	} + +	// Set the IRQ exception vector +	AT91F_AIC_SetExceptionVector((unsigned int *) 0x18, IrqHandler); +	// Set the Fast Interrupt exception vector +	AT91F_AIC_SetExceptionVector((unsigned int *) 0x1C, FiqHandler); + +	pAic->AIC_SPU = (unsigned int) SpuriousHandler; +	pAic->AIC_DCR = protectMode; +} + +//*---------------------------------------------------------------------------- +//* \fn    AT91F_PDC_Open +//* \brief Open PDC: disable TX and RX reset transfer descriptors, re-enable RX and TX +//*---------------------------------------------------------------------------- +void AT91F_PDC_Open(AT91PS_PDC pPDC)       // \arg pointer to a PDC controller +{ +    //* Disable the RX and TX PDC transfer requests +	AT91F_PDC_DisableRx(pPDC); +	AT91F_PDC_DisableTx(pPDC); + +	//* Reset all Counter register Next buffer first +	AT91F_PDC_SetNextTx(pPDC, NULL, 0); +	AT91F_PDC_SetNextRx(pPDC, NULL, 0); +	AT91F_PDC_SetTx(pPDC, NULL, 0); +	AT91F_PDC_SetRx(pPDC, NULL, 0); + +    //* Enable the RX and TX PDC transfer requests +	AT91F_PDC_EnableRx(pPDC); +	AT91F_PDC_EnableTx(pPDC); +} + +//*---------------------------------------------------------------------------- +//* \fn    AT91F_PDC_Close +//* \brief Close PDC: disable TX and RX reset transfer descriptors +//*---------------------------------------------------------------------------- +void AT91F_PDC_Close(AT91PS_PDC pPDC)       // \arg pointer to a PDC controller +{ +    //* Disable the RX and TX PDC transfer requests +	AT91F_PDC_DisableRx(pPDC); +	AT91F_PDC_DisableTx(pPDC); + +	//* Reset all Counter register Next buffer first +	AT91F_PDC_SetNextTx(pPDC, NULL, 0); +	AT91F_PDC_SetNextRx(pPDC, NULL, 0); +	AT91F_PDC_SetTx(pPDC, NULL, 0); +	AT91F_PDC_SetRx(pPDC, NULL, 0); + +} + +//*---------------------------------------------------------------------------- +//* \fn    AT91F_PDC_SendFrame +//* \brief Close PDC: disable TX and RX reset transfer descriptors +//*---------------------------------------------------------------------------- +unsigned int AT91F_PDC_SendFrame( +	AT91PS_PDC pPDC, +	const unsigned char *pBuffer, +	unsigned int szBuffer, +	const unsigned char *pNextBuffer, +	unsigned int szNextBuffer ) +{ +	if (AT91F_PDC_IsTxEmpty(pPDC)) { +		//* Buffer and next buffer can be initialized +		AT91F_PDC_SetTx(pPDC, pBuffer, szBuffer); +		AT91F_PDC_SetNextTx(pPDC, pNextBuffer, szNextBuffer); +		return 2; +	} +	else if (AT91F_PDC_IsNextTxEmpty(pPDC)) { +		//* Only one buffer can be initialized +		AT91F_PDC_SetNextTx(pPDC, pBuffer, szBuffer); +		return 1; +	} +	else { +		//* All buffer are in use... +		return 0; +	} +} + +//*---------------------------------------------------------------------------- +//* \fn    AT91F_PDC_ReceiveFrame +//* \brief Close PDC: disable TX and RX reset transfer descriptors +//*---------------------------------------------------------------------------- +unsigned int AT91F_PDC_ReceiveFrame ( +	AT91PS_PDC pPDC, +	unsigned char *pBuffer, +	unsigned int szBuffer, +	unsigned char *pNextBuffer, +	unsigned int szNextBuffer ) +{ +	if (AT91F_PDC_IsRxEmpty(pPDC)) { +		//* Buffer and next buffer can be initialized +		AT91F_PDC_SetRx(pPDC, pBuffer, szBuffer); +		AT91F_PDC_SetNextRx(pPDC, pNextBuffer, szNextBuffer); +		return 2; +	} +	else if (AT91F_PDC_IsNextRxEmpty(pPDC)) { +		//* Only one buffer can be initialized +		AT91F_PDC_SetNextRx(pPDC, pBuffer, szBuffer); +		return 1; +	} +	else { +		//* All buffer are in use... +		return 0; +	} +} + +//*------------------------------------------------------------------------------ +//* \fn    AT91F_PMC_GetMasterClock +//* \brief Return master clock in Hz which correponds to processor clock for ARM7 +//*------------------------------------------------------------------------------ +unsigned int AT91F_PMC_GetMasterClock ( +	AT91PS_PMC pPMC, // \arg pointer to PMC controller +	AT91PS_CKGR pCKGR, // \arg pointer to CKGR controller +	unsigned int slowClock)  // \arg slowClock in Hz +{ +	unsigned int reg = pPMC->PMC_MCKR; +	unsigned int prescaler = (1 << ((reg & AT91C_PMC_PRES) >> 2)); +	unsigned int pllDivider, pllMultiplier; + +	switch (reg & AT91C_PMC_CSS) { +		case AT91C_PMC_CSS_SLOW_CLK: // Slow clock selected +			return slowClock / prescaler; +		case AT91C_PMC_CSS_MAIN_CLK: // Main clock is selected +			return AT91F_CKGR_GetMainClock(pCKGR, slowClock) / prescaler; +		case AT91C_PMC_CSS_PLL_CLK: // PLLB clock is selected +			reg = pCKGR->CKGR_PLLR; +			pllDivider    = (reg  & AT91C_CKGR_DIV); +			pllMultiplier = ((reg  & AT91C_CKGR_MUL) >> 16) + 1; +			return AT91F_CKGR_GetMainClock(pCKGR, slowClock) / pllDivider * pllMultiplier / prescaler; +	} +	return 0; +} + +//*-------------------------------------------------------------------------------------- +//* \fn     AT91F_RTT_ReadValue() +//* \brief  Read the RTT value +//*-------------------------------------------------------------------------------------- +unsigned int AT91F_RTTReadValue(AT91PS_RTTC pRTTC) +{ +        register volatile unsigned int val1,val2; +	do +	{ +		val1 = pRTTC->RTTC_RTVR; +		val2 = pRTTC->RTTC_RTVR; +	}	 +	while(val1 != val2); +	return(val1); +} + +//*---------------------------------------------------------------------------- +//* \fn    AT91F_SPI_Close +//* \brief Close SPI: disable IT disable transfert, close PDC +//*---------------------------------------------------------------------------- +void AT91F_SPI_Close(AT91PS_SPI pSPI)     // \arg pointer to a SPI controller +{ +    //* Reset all the Chip Select register +    pSPI->SPI_CSR[0] = 0 ; +    pSPI->SPI_CSR[1] = 0 ; +    pSPI->SPI_CSR[2] = 0 ; +    pSPI->SPI_CSR[3] = 0 ; + +    //* Reset the SPI mode +    pSPI->SPI_MR = 0  ; + +    //* Disable all interrupts +    pSPI->SPI_IDR = 0xFFFFFFFF ; + +    //* Abort the Peripheral Data Transfers +    AT91F_PDC_Close((AT91PS_PDC) &(pSPI->SPI_RPR)); + +    //* Disable receiver and transmitter and stop any activity immediately +    pSPI->SPI_CR = AT91C_SPI_SPIDIS; +} + +//*---------------------------------------------------------------------------- +//* \fn    AT91F_ADC_CfgTimings +//* \brief Configure the different necessary timings of the ADC controller +//*---------------------------------------------------------------------------- +void AT91F_ADC_CfgTimings ( +	AT91PS_ADC pADC, // pointer to a ADC controller +	unsigned int mck_clock, // in MHz  +	unsigned int adc_clock, // in MHz  +	unsigned int startup_time, // in us  +	unsigned int sample_and_hold_time)	// in ns   +{ +	unsigned int prescal,startup,shtim; +	 +	prescal = mck_clock/(2*adc_clock) - 1; +	startup = adc_clock*startup_time/8 - 1; +	shtim = adc_clock*sample_and_hold_time/1000 - 1; +	 +	//* Write to the MR register +	pADC->ADC_MR = ( (prescal<<8) & AT91C_ADC_PRESCAL) | ( (startup<<16) & AT91C_ADC_STARTUP) | ( (shtim<<24) & AT91C_ADC_SHTIM); +} + +//*---------------------------------------------------------------------------- +//* \fn    AT91F_SSC_SetBaudrate +//* \brief Set the baudrate according to the CPU clock +//*---------------------------------------------------------------------------- +void AT91F_SSC_SetBaudrate ( +        AT91PS_SSC pSSC,        // \arg pointer to a SSC controller +        unsigned int mainClock, // \arg peripheral clock +        unsigned int speed)     // \arg SSC baudrate +{ +        unsigned int baud_value; +        //* Define the baud rate divisor register +        if (speed == 0) +           baud_value = 0; +        else +        { +           baud_value = (unsigned int) (mainClock * 10)/(2*speed); +           if ((baud_value % 10) >= 5) +                  baud_value = (baud_value / 10) + 1; +           else +                  baud_value /= 10; +        } + +        pSSC->SSC_CMR = baud_value; +} + +//*---------------------------------------------------------------------------- +//* \fn    AT91F_SSC_Configure +//* \brief Configure SSC +//*---------------------------------------------------------------------------- +void AT91F_SSC_Configure ( +             AT91PS_SSC pSSC,          // \arg pointer to a SSC controller +             unsigned int syst_clock,  // \arg System Clock Frequency +             unsigned int baud_rate,   // \arg Expected Baud Rate Frequency +             unsigned int clock_rx,    // \arg Receiver Clock Parameters +             unsigned int mode_rx,     // \arg mode Register to be programmed +             unsigned int clock_tx,    // \arg Transmitter Clock Parameters +             unsigned int mode_tx)     // \arg mode Register to be programmed +{ +    //* Disable interrupts +	pSSC->SSC_IDR = (unsigned int) -1; + +    //* Reset receiver and transmitter +	pSSC->SSC_CR = AT91C_SSC_SWRST | AT91C_SSC_RXDIS | AT91C_SSC_TXDIS ; + +    //* Define the Clock Mode Register +	AT91F_SSC_SetBaudrate(pSSC, syst_clock, baud_rate); + +     //* Write the Receive Clock Mode Register +	pSSC->SSC_RCMR =  clock_rx; + +     //* Write the Transmit Clock Mode Register +	pSSC->SSC_TCMR =  clock_tx; + +     //* Write the Receive Frame Mode Register +	pSSC->SSC_RFMR =  mode_rx; + +     //* Write the Transmit Frame Mode Register +	pSSC->SSC_TFMR =  mode_tx; + +    //* Clear Transmit and Receive Counters +	AT91F_PDC_Open((AT91PS_PDC) &(pSSC->SSC_RPR)); + + +} + +//*---------------------------------------------------------------------------- +//* \fn    AT91F_US_Configure +//* \brief Configure USART +//*---------------------------------------------------------------------------- +void AT91F_US_Configure ( +	AT91PS_USART pUSART,     // \arg pointer to a USART controller +	unsigned int mainClock,  // \arg peripheral clock +	unsigned int mode ,      // \arg mode Register to be programmed +	unsigned int baudRate ,  // \arg baudrate to be programmed +	unsigned int timeguard ) // \arg timeguard to be programmed +{ +    //* Disable interrupts +    pUSART->US_IDR = (unsigned int) -1; + +    //* Reset receiver and transmitter +    pUSART->US_CR = AT91C_US_RSTRX | AT91C_US_RSTTX | AT91C_US_RXDIS | AT91C_US_TXDIS ; + +	//* Define the baud rate divisor register +	AT91F_US_SetBaudrate(pUSART, mainClock, baudRate); + +	//* Write the Timeguard Register +	AT91F_US_SetTimeguard(pUSART, timeguard); + +    //* Clear Transmit and Receive Counters +    AT91F_PDC_Open((AT91PS_PDC) &(pUSART->US_RPR)); + +    //* Define the USART mode +    pUSART->US_MR = mode  ; + +} + +//*---------------------------------------------------------------------------- +//* \fn    AT91F_US_Close +//* \brief Close USART: disable IT disable receiver and transmitter, close PDC +//*---------------------------------------------------------------------------- +void AT91F_US_Close(AT91PS_USART pUSART)     // \arg pointer to a USART controller +{ +    //* Reset the baud rate divisor register +    pUSART->US_BRGR = 0 ; + +    //* Reset the USART mode +    pUSART->US_MR = 0  ; + +    //* Reset the Timeguard Register +    pUSART->US_TTGR = 0; + +    //* Disable all interrupts +    pUSART->US_IDR = 0xFFFFFFFF ; + +    //* Abort the Peripheral Data Transfers +    AT91F_PDC_Close((AT91PS_PDC) &(pUSART->US_RPR)); + +    //* Disable receiver and transmitter and stop any activity immediately +    pUSART->US_CR = AT91C_US_TXDIS | AT91C_US_RXDIS | AT91C_US_RSTTX | AT91C_US_RSTRX ; +} + + diff --git a/firmware/lib/memcpy.S b/firmware/lib/memcpy.S new file mode 100644 index 0000000..2bbd569 --- /dev/null +++ b/firmware/lib/memcpy.S @@ -0,0 +1,59 @@ +/* + *  linux/arch/arm/lib/memcpy.S + * + *  Author:	Nicolas Pitre + *  Created:	Sep 28, 2005 + *  Copyright:	MontaVista Software, Inc. + * + *  This program is free software; you can redistribute it and/or modify + *  it under the terms of the GNU General Public License version 2 as + *  published by the Free Software Foundation. + */ + +#include <asm/linkage.h> +#include <asm/assembler.h> + +	.macro ldr1w ptr reg abort +	ldr \reg, [\ptr], #4 +	.endm + +	.macro ldr4w ptr reg1 reg2 reg3 reg4 abort +	ldmia \ptr!, {\reg1, \reg2, \reg3, \reg4} +	.endm + +	.macro ldr8w ptr reg1 reg2 reg3 reg4 reg5 reg6 reg7 reg8 abort +	ldmia \ptr!, {\reg1, \reg2, \reg3, \reg4, \reg5, \reg6, \reg7, \reg8} +	.endm + +	.macro ldr1b ptr reg cond=al abort +	ldr\cond\()b \reg, [\ptr], #1 +	.endm + +	.macro str1w ptr reg abort +	str \reg, [\ptr], #4 +	.endm + +	.macro str8w ptr reg1 reg2 reg3 reg4 reg5 reg6 reg7 reg8 abort +	stmia \ptr!, {\reg1, \reg2, \reg3, \reg4, \reg5, \reg6, \reg7, \reg8} +	.endm + +	.macro str1b ptr reg cond=al abort +	str\cond\()b \reg, [\ptr], #1 +	.endm + +	.macro enter reg1 reg2 +	stmdb sp!, {r0, \reg1, \reg2} +	.endm + +	.macro exit reg1 reg2 +	ldmfd sp!, {r0, \reg1, \reg2} +	.endm + +	.text + +/* Prototype: void *memcpy(void *dest, const void *src, size_t n); */ + +ENTRY(memcpy) + +#include "copy_template.S" + diff --git a/firmware/lib/memset.S b/firmware/lib/memset.S new file mode 100644 index 0000000..04e254a --- /dev/null +++ b/firmware/lib/memset.S @@ -0,0 +1,80 @@ +/* + *  linux/arch/arm/lib/memset.S + * + *  Copyright (C) 1995-2000 Russell King + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License version 2 as + * published by the Free Software Foundation. + * + *  ASM optimised string functions + */ +#include <asm/linkage.h> +#include <asm/assembler.h> + +	.text +	.align	5 +	.word	0 + +1:	subs	r2, r2, #4		@ 1 do we have enough +	blt	5f			@ 1 bytes to align with? +	cmp	r3, #2			@ 1 +	strltb	r1, [r0], #1		@ 1 +	strleb	r1, [r0], #1		@ 1 +	strb	r1, [r0], #1		@ 1 +	add	r2, r2, r3		@ 1 (r2 = r2 - (4 - r3)) +/* + * The pointer is now aligned and the length is adjusted.  Try doing the + * memzero again. + */ + +ENTRY(memset) +	ands	r3, r0, #3		@ 1 unaligned? +	bne	1b			@ 1 +/* + * we know that the pointer in r0 is aligned to a word boundary. + */ +	orr	r1, r1, r1, lsl #8 +	orr	r1, r1, r1, lsl #16 +	mov	r3, r1 +	cmp	r2, #16 +	blt	4f +/* + * We need an extra register for this loop - save the return address and + * use the LR + */ +	str	lr, [sp, #-4]! +	mov	ip, r1 +	mov	lr, r1 + +2:	subs	r2, r2, #64 +	stmgeia	r0!, {r1, r3, ip, lr}	@ 64 bytes at a time. +	stmgeia	r0!, {r1, r3, ip, lr} +	stmgeia	r0!, {r1, r3, ip, lr} +	stmgeia	r0!, {r1, r3, ip, lr} +	bgt	2b +	LOADREGS(eqfd, sp!, {pc})	@ Now <64 bytes to go. +/* + * No need to correct the count; we're only testing bits from now on + */ +	tst	r2, #32 +	stmneia	r0!, {r1, r3, ip, lr} +	stmneia	r0!, {r1, r3, ip, lr} +	tst	r2, #16 +	stmneia	r0!, {r1, r3, ip, lr} +	ldr	lr, [sp], #4 + +4:	tst	r2, #8 +	stmneia	r0!, {r1, r3} +	tst	r2, #4 +	strne	r1, [r0], #4 +/* + * When we get here, we've got less than 4 bytes to zero.  We + * may have an unaligned pointer as well. + */ +5:	tst	r2, #2 +	strneb	r1, [r0], #1 +	strneb	r1, [r0], #1 +	tst	r2, #1 +	strneb	r1, [r0], #1 +	RETINSTR(mov,pc,lr) diff --git a/firmware/lib/setbit.S b/firmware/lib/setbit.S new file mode 100644 index 0000000..9009bc1 --- /dev/null +++ b/firmware/lib/setbit.S @@ -0,0 +1,22 @@ +/* + *  linux/arch/arm/lib/setbit.S + * + *  Copyright (C) 1995-1996 Russell King + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License version 2 as + * published by the Free Software Foundation. + */ +#include <asm/linkage.h> +#include <asm/assembler.h> +#include "bitops.h" +		.text + +/* + * Purpose  : Function to set a bit + * Prototype: int set_bit(int bit, void *addr) + */ +ENTRY(_set_bit_be) +		eor	r0, r0, #0x18		@ big endian byte ordering +ENTRY(_set_bit_le) +	bitop	orr diff --git a/firmware/lib/string.c b/firmware/lib/string.c new file mode 100644 index 0000000..4158bca --- /dev/null +++ b/firmware/lib/string.c @@ -0,0 +1,41 @@ +/* + *  linux/lib/string.c + * + *  Copyright (C) 1991, 1992  Linus Torvalds + */ + +/* + * stupid library routines.. The optimized versions should generally be found + * as inline code in <asm-xx/string.h> + * + * These are buggy as well.. + * + * * Fri Jun 25 1999, Ingo Oeser <ioe@informatik.tu-chemnitz.de> + * -  Added strsep() which will replace strtok() soon (because strsep() is + *    reentrant and should be faster). Use only strsep() in new code, please. + * + * * Sat Feb 09 2002, Jason Thomas <jason@topic.com.au>, + *                    Matthew Hawkins <matt@mh.dropbear.id.au> + * -  Kissed strtok() goodbye + */ + +#include <sys/types.h> +#include <string.h> +#include <asm/ctype.h> + + +#ifndef __HAVE_ARCH_STRNLEN +/** + * strnlen - Find the length of a length-limited string + * @s: The string to be sized + * @count: The maximum number of bytes to search + */ +size_t strnlen(const char *s, size_t count) +{ +	const char *sc; + +	for (sc = s; count-- && *sc != '\0'; ++sc) +		/* nothing */; +	return sc - s; +} +#endif diff --git a/firmware/lib/testchangebit.S b/firmware/lib/testchangebit.S new file mode 100644 index 0000000..37c303e --- /dev/null +++ b/firmware/lib/testchangebit.S @@ -0,0 +1,18 @@ +/* + *  linux/arch/arm/lib/testchangebit.S + * + *  Copyright (C) 1995-1996 Russell King + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License version 2 as + * published by the Free Software Foundation. + */ +#include <asm/linkage.h> +#include <asm/assembler.h> +#include "bitops.h" +                .text + +ENTRY(_test_and_change_bit_be) +		eor	r0, r0, #0x18		@ big endian byte ordering +ENTRY(_test_and_change_bit_le) +	testop	eor, strb diff --git a/firmware/lib/testclearbit.S b/firmware/lib/testclearbit.S new file mode 100644 index 0000000..985c399 --- /dev/null +++ b/firmware/lib/testclearbit.S @@ -0,0 +1,18 @@ +/* + *  linux/arch/arm/lib/testclearbit.S + * + *  Copyright (C) 1995-1996 Russell King + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License version 2 as + * published by the Free Software Foundation. + */ +#include <asm/linkage.h> +#include <asm/assembler.h> +#include "bitops.h" +                .text + +ENTRY(_test_and_clear_bit_be) +		eor	r0, r0, #0x18		@ big endian byte ordering +ENTRY(_test_and_clear_bit_le) +	testop	bicne, strneb diff --git a/firmware/lib/testsetbit.S b/firmware/lib/testsetbit.S new file mode 100644 index 0000000..4a8a164 --- /dev/null +++ b/firmware/lib/testsetbit.S @@ -0,0 +1,18 @@ +/* + *  linux/arch/arm/lib/testsetbit.S + * + *  Copyright (C) 1995-1996 Russell King + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License version 2 as + * published by the Free Software Foundation. + */ +#include <asm/linkage.h> +#include <asm/assembler.h> +#include "bitops.h" +                .text + +ENTRY(_test_and_set_bit_be) +		eor	r0, r0, #0x18		@ big endian byte ordering +ENTRY(_test_and_set_bit_le) +	testop	orreq, streqb diff --git a/firmware/lib/vsprintf.c b/firmware/lib/vsprintf.c new file mode 100644 index 0000000..9eff70b --- /dev/null +++ b/firmware/lib/vsprintf.c @@ -0,0 +1,830 @@ +/* + *  linux/lib/vsprintf.c + * + *  Copyright (C) 1991, 1992  Linus Torvalds + */ + +/* vsprintf.c -- Lars Wirzenius & Linus Torvalds. */ +/* + * Wirzenius wrote this portably, Torvalds fucked it up :-) + */ + +/*  + * Fri Jul 13 2001 Crutcher Dunnavant <crutcher+kernel@datastacks.com> + * - changed to provide snprintf and vsnprintf functions + * So Feb  1 16:51:32 CET 2004 Juergen Quade <quade@hsnr.de> + * - scnprintf and vscnprintf + */ + +#include <stdarg.h> +#include <sys/types.h> +#include <string.h> +#include <asm/ctype.h> + +#include <asm/div64.h> + +/** + * simple_strtoul - convert a string to an unsigned long + * @cp: The start of the string + * @endp: A pointer to the end of the parsed string will be placed here + * @base: The number base to use + */ +unsigned long simple_strtoul(const char *cp,char **endp,unsigned int base) +{ +	unsigned long result = 0,value; + +	if (!base) { +		base = 10; +		if (*cp == '0') { +			base = 8; +			cp++; +			if ((toupper(*cp) == 'X') && isxdigit(cp[1])) { +				cp++; +				base = 16; +			} +		} +	} else if (base == 16) { +		if (cp[0] == '0' && toupper(cp[1]) == 'X') +			cp += 2; +	} +	while (isxdigit(*cp) && +	       (value = isdigit(*cp) ? *cp-'0' : toupper(*cp)-'A'+10) < base) { +		result = result*base + value; +		cp++; +	} +	if (endp) +		*endp = (char *)cp; +	return result; +} + + +/** + * simple_strtol - convert a string to a signed long + * @cp: The start of the string + * @endp: A pointer to the end of the parsed string will be placed here + * @base: The number base to use + */ +long simple_strtol(const char *cp,char **endp,unsigned int base) +{ +	if(*cp=='-') +		return -simple_strtoul(cp+1,endp,base); +	return simple_strtoul(cp,endp,base); +} + + +/** + * simple_strtoull - convert a string to an unsigned long long + * @cp: The start of the string + * @endp: A pointer to the end of the parsed string will be placed here + * @base: The number base to use + */ +unsigned long long simple_strtoull(const char *cp,char **endp,unsigned int base) +{ +	unsigned long long result = 0,value; + +	if (!base) { +		base = 10; +		if (*cp == '0') { +			base = 8; +			cp++; +			if ((toupper(*cp) == 'X') && isxdigit(cp[1])) { +				cp++; +				base = 16; +			} +		} +	} else if (base == 16) { +		if (cp[0] == '0' && toupper(cp[1]) == 'X') +			cp += 2; +	} +	while (isxdigit(*cp) && (value = isdigit(*cp) ? *cp-'0' : (islower(*cp) +	    ? toupper(*cp) : *cp)-'A'+10) < base) { +		result = result*base + value; +		cp++; +	} +	if (endp) +		*endp = (char *)cp; +	return result; +} + + +/** + * simple_strtoll - convert a string to a signed long long + * @cp: The start of the string + * @endp: A pointer to the end of the parsed string will be placed here + * @base: The number base to use + */ +long long simple_strtoll(const char *cp,char **endp,unsigned int base) +{ +	if(*cp=='-') +		return -simple_strtoull(cp+1,endp,base); +	return simple_strtoull(cp,endp,base); +} + +static int skip_atoi(const char **s) +{ +	int i=0; + +	while (isdigit(**s)) +		i = i*10 + *((*s)++) - '0'; +	return i; +} + +#define ZEROPAD	1		/* pad with zero */ +#define SIGN	2		/* unsigned/signed long */ +#define PLUS	4		/* show plus */ +#define SPACE	8		/* space if plus */ +#define LEFT	16		/* left justified */ +#define SPECIAL	32		/* 0x */ +#define LARGE	64		/* use 'ABCDEF' instead of 'abcdef' */ + +static char * number(char * buf, char * end, unsigned long long num, int base, int size, int precision, int type) +{ +	char c,sign,tmp[66]; +	const char *digits; +	static const char small_digits[] = "0123456789abcdefghijklmnopqrstuvwxyz"; +	static const char large_digits[] = "0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZ"; +	int i; + +	digits = (type & LARGE) ? large_digits : small_digits; +	if (type & LEFT) +		type &= ~ZEROPAD; +	if (base < 2 || base > 36) +		return NULL; +	c = (type & ZEROPAD) ? '0' : ' '; +	sign = 0; +	if (type & SIGN) { +		if ((signed long long) num < 0) { +			sign = '-'; +			num = - (signed long long) num; +			size--; +		} else if (type & PLUS) { +			sign = '+'; +			size--; +		} else if (type & SPACE) { +			sign = ' '; +			size--; +		} +	} +	if (type & SPECIAL) { +		if (base == 16) +			size -= 2; +		else if (base == 8) +			size--; +	} +	i = 0; +	if (num == 0) +		tmp[i++]='0'; +	else while (num != 0) +		tmp[i++] = digits[do_div(num,base)]; +	if (i > precision) +		precision = i; +	size -= precision; +	if (!(type&(ZEROPAD+LEFT))) { +		while(size-->0) { +			if (buf <= end) +				*buf = ' '; +			++buf; +		} +	} +	if (sign) { +		if (buf <= end) +			*buf = sign; +		++buf; +	} +	if (type & SPECIAL) { +		if (base==8) { +			if (buf <= end) +				*buf = '0'; +			++buf; +		} else if (base==16) { +			if (buf <= end) +				*buf = '0'; +			++buf; +			if (buf <= end) +				*buf = digits[33]; +			++buf; +		} +	} +	if (!(type & LEFT)) { +		while (size-- > 0) { +			if (buf <= end) +				*buf = c; +			++buf; +		} +	} +	while (i < precision--) { +		if (buf <= end) +			*buf = '0'; +		++buf; +	} +	while (i-- > 0) { +		if (buf <= end) +			*buf = tmp[i]; +		++buf; +	} +	while (size-- > 0) { +		if (buf <= end) +			*buf = ' '; +		++buf; +	} +	return buf; +} + +/** + * vsnprintf - Format a string and place it in a buffer + * @buf: The buffer to place the result into + * @size: The size of the buffer, including the trailing null space + * @fmt: The format string to use + * @args: Arguments for the format string + * + * The return value is the number of characters which would + * be generated for the given input, excluding the trailing + * '\0', as per ISO C99. If you want to have the exact + * number of characters written into @buf as return value + * (not including the trailing '\0'), use vscnprintf. If the + * return is greater than or equal to @size, the resulting + * string is truncated. + * + * Call this function if you are already dealing with a va_list. + * You probably want snprintf instead. + */ +int vsnprintf(char *buf, size_t size, const char *fmt, va_list args) +{ +	int len; +	unsigned long long num; +	int i, base; +	char *str, *end, c; +	const char *s; + +	int flags;		/* flags to number() */ + +	int field_width;	/* width of output field */ +	int precision;		/* min. # of digits for integers; max +				   number of chars for from string */ +	int qualifier;		/* 'h', 'l', or 'L' for integer fields */ +				/* 'z' support added 23/7/1999 S.H.    */ +				/* 'z' changed to 'Z' --davidm 1/25/99 */ +				/* 't' added for ptrdiff_t */ + +	/* Reject out-of-range values early */ +	if ((int) size < 0) { +		return 0; +	} + +	str = buf; +	end = buf + size - 1; + +	if (end < buf - 1) { +		end = ((void *) -1); +		size = end - buf + 1; +	} + +	for (; *fmt ; ++fmt) { +		if (*fmt != '%') { +			if (str <= end) +				*str = *fmt; +			++str; +			continue; +		} + +		/* process flags */ +		flags = 0; +		repeat: +			++fmt;		/* this also skips first '%' */ +			switch (*fmt) { +				case '-': flags |= LEFT; goto repeat; +				case '+': flags |= PLUS; goto repeat; +				case ' ': flags |= SPACE; goto repeat; +				case '#': flags |= SPECIAL; goto repeat; +				case '0': flags |= ZEROPAD; goto repeat; +			} + +		/* get field width */ +		field_width = -1; +		if (isdigit(*fmt)) +			field_width = skip_atoi(&fmt); +		else if (*fmt == '*') { +			++fmt; +			/* it's the next argument */ +			field_width = va_arg(args, int); +			if (field_width < 0) { +				field_width = -field_width; +				flags |= LEFT; +			} +		} + +		/* get the precision */ +		precision = -1; +		if (*fmt == '.') { +			++fmt;	 +			if (isdigit(*fmt)) +				precision = skip_atoi(&fmt); +			else if (*fmt == '*') { +				++fmt; +				/* it's the next argument */ +				precision = va_arg(args, int); +			} +			if (precision < 0) +				precision = 0; +		} + +		/* get the conversion qualifier */ +		qualifier = -1; +		if (*fmt == 'h' || *fmt == 'l' || *fmt == 'L' || +		    *fmt =='Z' || *fmt == 'z' || *fmt == 't') { +			qualifier = *fmt; +			++fmt; +			if (qualifier == 'l' && *fmt == 'l') { +				qualifier = 'L'; +				++fmt; +			} +		} + +		/* default base */ +		base = 10; + +		switch (*fmt) { +			case 'c': +				if (!(flags & LEFT)) { +					while (--field_width > 0) { +						if (str <= end) +							*str = ' '; +						++str; +					} +				} +				c = (unsigned char) va_arg(args, int); +				if (str <= end) +					*str = c; +				++str; +				while (--field_width > 0) { +					if (str <= end) +						*str = ' '; +					++str; +				} +				continue; + +			case 's': +				s = va_arg(args, char *); + +				len = strnlen(s, precision); + +				if (!(flags & LEFT)) { +					while (len < field_width--) { +						if (str <= end) +							*str = ' '; +						++str; +					} +				} +				for (i = 0; i < len; ++i) { +					if (str <= end) +						*str = *s; +					++str; ++s; +				} +				while (len < field_width--) { +					if (str <= end) +						*str = ' '; +					++str; +				} +				continue; + +			case 'p': +				if (field_width == -1) { +					field_width = 2*sizeof(void *); +					flags |= ZEROPAD; +				} +				str = number(str, end, +						(unsigned long) va_arg(args, void *), +						16, field_width, precision, flags); +				continue; + + +			case 'n': +				/* FIXME: +				* What does C99 say about the overflow case here? */ +				if (qualifier == 'l') { +					long * ip = va_arg(args, long *); +					*ip = (str - buf); +				} else if (qualifier == 'Z' || qualifier == 'z') { +					size_t * ip = va_arg(args, size_t *); +					*ip = (str - buf); +				} else { +					int * ip = va_arg(args, int *); +					*ip = (str - buf); +				} +				continue; + +			case '%': +				if (str <= end) +					*str = '%'; +				++str; +				continue; + +				/* integer number formats - set up the flags and "break" */ +			case 'o': +				base = 8; +				break; + +			case 'X': +				flags |= LARGE; +			case 'x': +				base = 16; +				break; + +			case 'd': +			case 'i': +				flags |= SIGN; +			case 'u': +				break; + +			default: +				if (str <= end) +					*str = '%'; +				++str; +				if (*fmt) { +					if (str <= end) +						*str = *fmt; +					++str; +				} else { +					--fmt; +				} +				continue; +		} +		if (qualifier == 'L') +			num = va_arg(args, long long); +		else if (qualifier == 'l') { +			num = va_arg(args, unsigned long); +			if (flags & SIGN) +				num = (signed long) num; +		} else if (qualifier == 'Z' || qualifier == 'z') { +			num = va_arg(args, size_t); +		} else if (qualifier == 't') { +			num = va_arg(args, ptrdiff_t); +		} else if (qualifier == 'h') { +			num = (unsigned short) va_arg(args, int); +			if (flags & SIGN) +				num = (signed short) num; +		} else { +			num = va_arg(args, unsigned int); +			if (flags & SIGN) +				num = (signed int) num; +		} +		str = number(str, end, num, base, +				field_width, precision, flags); +	} +	if (str <= end) +		*str = '\0'; +	else if (size > 0) +		/* don't write out a null byte if the buf size is zero */ +		*end = '\0'; +	/* the trailing null byte doesn't count towards the total +	* ++str; +	*/ +	return str-buf; +} + + +/** + * vscnprintf - Format a string and place it in a buffer + * @buf: The buffer to place the result into + * @size: The size of the buffer, including the trailing null space + * @fmt: The format string to use + * @args: Arguments for the format string + * + * The return value is the number of characters which have been written into + * the @buf not including the trailing '\0'. If @size is <= 0 the function + * returns 0. + * + * Call this function if you are already dealing with a va_list. + * You probably want scnprintf instead. + */ +int vscnprintf(char *buf, size_t size, const char *fmt, va_list args) +{ +	int i; + +	i=vsnprintf(buf,size,fmt,args); +	return (i >= size) ? (size - 1) : i; +} + + +/** + * snprintf - Format a string and place it in a buffer + * @buf: The buffer to place the result into + * @size: The size of the buffer, including the trailing null space + * @fmt: The format string to use + * @...: Arguments for the format string + * + * The return value is the number of characters which would be + * generated for the given input, excluding the trailing null, + * as per ISO C99.  If the return is greater than or equal to + * @size, the resulting string is truncated. + */ +int snprintf(char * buf, size_t size, const char *fmt, ...) +{ +	va_list args; +	int i; + +	va_start(args, fmt); +	i=vsnprintf(buf,size,fmt,args); +	va_end(args); +	return i; +} + + +/** + * scnprintf - Format a string and place it in a buffer + * @buf: The buffer to place the result into + * @size: The size of the buffer, including the trailing null space + * @fmt: The format string to use + * @...: Arguments for the format string + * + * The return value is the number of characters written into @buf not including + * the trailing '\0'. If @size is <= 0 the function returns 0. If the return is + * greater than or equal to @size, the resulting string is truncated. + */ + +int scnprintf(char * buf, size_t size, const char *fmt, ...) +{ +	va_list args; +	int i; + +	va_start(args, fmt); +	i = vsnprintf(buf, size, fmt, args); +	va_end(args); +	return (i >= size) ? (size - 1) : i; +} + +/** + * vsprintf - Format a string and place it in a buffer + * @buf: The buffer to place the result into + * @fmt: The format string to use + * @args: Arguments for the format string + * + * The function returns the number of characters written + * into @buf. Use vsnprintf or vscnprintf in order to avoid + * buffer overflows. + * + * Call this function if you are already dealing with a va_list. + * You probably want sprintf instead. + */ +int vsprintf(char *buf, const char *fmt, va_list args) +{ +	return vsnprintf(buf, INT_MAX, fmt, args); +} + + +/** + * sprintf - Format a string and place it in a buffer + * @buf: The buffer to place the result into + * @fmt: The format string to use + * @...: Arguments for the format string + * + * The function returns the number of characters written + * into @buf. Use snprintf or scnprintf in order to avoid + * buffer overflows. + */ +int sprintf(char * buf, const char *fmt, ...) +{ +	va_list args; +	int i; + +	va_start(args, fmt); +	i=vsnprintf(buf, INT_MAX, fmt, args); +	va_end(args); +	return i; +} + + +/** + * vsscanf - Unformat a buffer into a list of arguments + * @buf:	input buffer + * @fmt:	format of buffer + * @args:	arguments + */ +int vsscanf(const char * buf, const char * fmt, va_list args) +{ +	const char *str = buf; +	char *next; +	char digit; +	int num = 0; +	int qualifier; +	int base; +	int field_width; +	int is_sign = 0; + +	while(*fmt && *str) { +		/* skip any white space in format */ +		/* white space in format matchs any amount of +		 * white space, including none, in the input. +		 */ +		if (isspace(*fmt)) { +			while (isspace(*fmt)) +				++fmt; +			while (isspace(*str)) +				++str; +		} + +		/* anything that is not a conversion must match exactly */ +		if (*fmt != '%' && *fmt) { +			if (*fmt++ != *str++) +				break; +			continue; +		} + +		if (!*fmt) +			break; +		++fmt; +		 +		/* skip this conversion. +		 * advance both strings to next white space +		 */ +		if (*fmt == '*') { +			while (!isspace(*fmt) && *fmt) +				fmt++; +			while (!isspace(*str) && *str) +				str++; +			continue; +		} + +		/* get field width */ +		field_width = -1; +		if (isdigit(*fmt)) +			field_width = skip_atoi(&fmt); + +		/* get conversion qualifier */ +		qualifier = -1; +		if (*fmt == 'h' || *fmt == 'l' || *fmt == 'L' || +		    *fmt == 'Z' || *fmt == 'z') { +			qualifier = *fmt++; +			if (qualifier == *fmt) { +				if (qualifier == 'h') { +					qualifier = 'H'; +					fmt++; +				} else if (qualifier == 'l') { +					qualifier = 'L'; +					fmt++; +				} +			} +		} +		base = 10; +		is_sign = 0; + +		if (!*fmt || !*str) +			break; + +		switch(*fmt++) { +		case 'c': +		{ +			char *s = (char *) va_arg(args,char*); +			if (field_width == -1) +				field_width = 1; +			do { +				*s++ = *str++; +			} while (--field_width > 0 && *str); +			num++; +		} +		continue; +		case 's': +		{ +			char *s = (char *) va_arg(args, char *); +			if(field_width == -1) +				field_width = INT_MAX; +			/* first, skip leading white space in buffer */ +			while (isspace(*str)) +				str++; + +			/* now copy until next white space */ +			while (*str && !isspace(*str) && field_width--) { +				*s++ = *str++; +			} +			*s = '\0'; +			num++; +		} +		continue; +		case 'n': +			/* return number of characters read so far */ +		{ +			int *i = (int *)va_arg(args,int*); +			*i = str - buf; +		} +		continue; +		case 'o': +			base = 8; +			break; +		case 'x': +		case 'X': +			base = 16; +			break; +		case 'i': +                        base = 0; +		case 'd': +			is_sign = 1; +		case 'u': +			break; +		case '%': +			/* looking for '%' in str */ +			if (*str++ != '%')  +				return num; +			continue; +		default: +			/* invalid format; stop here */ +			return num; +		} + +		/* have some sort of integer conversion. +		 * first, skip white space in buffer. +		 */ +		while (isspace(*str)) +			str++; + +		digit = *str; +		if (is_sign && digit == '-') +			digit = *(str + 1); + +		if (!digit +                    || (base == 16 && !isxdigit(digit)) +                    || (base == 10 && !isdigit(digit)) +                    || (base == 8 && (!isdigit(digit) || digit > '7')) +                    || (base == 0 && !isdigit(digit))) +				break; + +		switch(qualifier) { +		case 'H':	/* that's 'hh' in format */ +			if (is_sign) { +				signed char *s = (signed char *) va_arg(args,signed char *); +				*s = (signed char) simple_strtol(str,&next,base); +			} else { +				unsigned char *s = (unsigned char *) va_arg(args, unsigned char *); +				*s = (unsigned char) simple_strtoul(str, &next, base); +			} +			break; +		case 'h': +			if (is_sign) { +				short *s = (short *) va_arg(args,short *); +				*s = (short) simple_strtol(str,&next,base); +			} else { +				unsigned short *s = (unsigned short *) va_arg(args, unsigned short *); +				*s = (unsigned short) simple_strtoul(str, &next, base); +			} +			break; +		case 'l': +			if (is_sign) { +				long *l = (long *) va_arg(args,long *); +				*l = simple_strtol(str,&next,base); +			} else { +				unsigned long *l = (unsigned long*) va_arg(args,unsigned long*); +				*l = simple_strtoul(str,&next,base); +			} +			break; +		case 'L': +			if (is_sign) { +				long long *l = (long long*) va_arg(args,long long *); +				*l = simple_strtoll(str,&next,base); +			} else { +				unsigned long long *l = (unsigned long long*) va_arg(args,unsigned long long*); +				*l = simple_strtoull(str,&next,base); +			} +			break; +		case 'Z': +		case 'z': +		{ +			size_t *s = (size_t*) va_arg(args,size_t*); +			*s = (size_t) simple_strtoul(str,&next,base); +		} +		break; +		default: +			if (is_sign) { +				int *i = (int *) va_arg(args, int*); +				*i = (int) simple_strtol(str,&next,base); +			} else { +				unsigned int *i = (unsigned int*) va_arg(args, unsigned int*); +				*i = (unsigned int) simple_strtoul(str,&next,base); +			} +			break; +		} +		num++; + +		if (!next) +			break; +		str = next; +	} +	return num; +} + + +/** + * sscanf - Unformat a buffer into a list of arguments + * @buf:	input buffer + * @fmt:	formatting of buffer + * @...:	resulting arguments + */ +int sscanf(const char * buf, const char * fmt, ...) +{ +	va_list args; +	int i; + +	va_start(args,fmt); +	i = vsscanf(buf,fmt,args); +	va_end(args); +	return i; +} + | 
