Donate to e Foundation | Murena handsets with /e/OS | Own a part of Murena! Learn more

Commit 9f825962 authored by David S. Miller's avatar David S. Miller
Browse files

sparc64: Niagara-4 bzero/memset, plus use MRU stores in page copy.



This adds optimized memset/bzero/page-clear routines for Niagara-4.

We basically can do what powerpc has been able to do for a decade (via
the "dcbz" instruction), which is use cache line clearing stores for
bzero and memsets with a 'c' argument of zero.

As long as we make the cache initializing store to each 32-byte
subblock of the L2 cache line, it works.

As with other Niagara-4 optimized routines, the key is to make sure to
avoid any usage of the %asi register, as reads and writes to it cost
at least 50 cycles.

For the user clear cases, we don't use these new routines, we use the
Niagara-1 variants instead.  Those have to use %asi in an unavoidable
way.

A Niagara-4 8K page clear costs just under 600 cycles.

Add definitions of the MRU variants of the cache initializing store
ASIs.  By default, cache initializing stores install the line as Least
Recently Used.  If we know we're going to use the data immediately
(which is true for page copies and clears) we can use the Most
Recently Used variant, to decrease the likelyhood of the lines being
evicted before they get used.

Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parent ffa9009c
Loading
Loading
Loading
Loading
+19 −0
Original line number Diff line number Diff line
@@ -270,9 +270,28 @@
#define ASI_BLK_INIT_QUAD_LDD_P	0xe2 /* (NG) init-store, twin load,
				      * primary, implicit
				      */
#define ASI_BLK_INIT_QUAD_LDD_S	0xe3 /* (NG) init-store, twin load,
				      * secondary, implicit
				      */
#define ASI_BLK_P		0xf0 /* Primary, blk ld/st		*/
#define ASI_BLK_S		0xf1 /* Secondary, blk ld/st		*/
#define ASI_ST_BLKINIT_MRU_P	0xf2 /* (NG4) init-store, twin load,
				      * Most-Recently-Used, primary,
				      * implicit
				      */
#define ASI_ST_BLKINIT_MRU_S	0xf2 /* (NG4) init-store, twin load,
				      * Most-Recently-Used, secondary,
				      * implicit
				      */
#define ASI_BLK_PL		0xf8 /* Primary, blk ld/st, little	*/
#define ASI_BLK_SL		0xf9 /* Secondary, blk ld/st, little	*/
#define ASI_ST_BLKINIT_MRU_PL	0xfa /* (NG4) init-store, twin load,
				      * Most-Recently-Used, primary,
				      * implicit, little-endian
				      */
#define ASI_ST_BLKINIT_MRU_SL	0xfb /* (NG4) init-store, twin load,
				      * Most-Recently-Used, secondary,
				      * implicit, little-endian
				      */

#endif /* _SPARC_ASI_H */
+1 −1
Original line number Diff line number Diff line
@@ -576,7 +576,7 @@ niagara_tlb_fixup:
niagara4_patch:
	call	niagara4_patch_copyops
	 nop
	call	niagara_patch_bzero
	call	niagara4_patch_bzero
	 nop
	call	niagara4_patch_pageops
	 nop
+1 −1
Original line number Diff line number Diff line
@@ -33,7 +33,7 @@ lib-$(CONFIG_SPARC64) += NG2memcpy.o NG2copy_from_user.o NG2copy_to_user.o
lib-$(CONFIG_SPARC64) +=  NG2patch.o

lib-$(CONFIG_SPARC64) += NG4memcpy.o NG4copy_from_user.o NG4copy_to_user.o
lib-$(CONFIG_SPARC64) +=  NG4patch.o NG4copy_page.o
lib-$(CONFIG_SPARC64) +=  NG4patch.o NG4copy_page.o NG4clear_page.o NG4memset.o

lib-$(CONFIG_SPARC64) += GENmemcpy.o GENcopy_from_user.o GENcopy_to_user.o
lib-$(CONFIG_SPARC64) += GENpatch.o GENpage.o GENbzero.o
+29 −0
Original line number Diff line number Diff line
/* NG4copy_page.S: Niagara-4 optimized clear page.
 *
 * Copyright (C) 2012 (davem@davemloft.net)
 */

#include <asm/asi.h>
#include <asm/page.h>

	.text

	.register	%g3, #scratch

	.align		32
	.globl		NG4clear_page
	.globl		NG4clear_user_page
NG4clear_page:		/* %o0=dest */
NG4clear_user_page:	/* %o0=dest, %o1=vaddr */
	set		PAGE_SIZE, %g7
	mov		0x20, %g3
1:	stxa		%g0, [%o0 + %g0] ASI_ST_BLKINIT_MRU_P
	subcc		%g7, 0x40, %g7
	stxa		%g0, [%o0 + %g3] ASI_ST_BLKINIT_MRU_P
	bne,pt		%xcc, 1b
	 add		%o0, 0x40, %o0
	membar		#StoreLoad|#StoreStore
	retl
	 nop
	.size		NG4clear_page,.-NG4clear_page
	.size		NG4clear_user_page,.-NG4clear_user_page
 No newline at end of file
+8 −8
Original line number Diff line number Diff line
@@ -30,25 +30,25 @@ NG4copy_user_page: /* %o0=dest, %o1=src, %o2=vaddr */
	ldx		[%o1 + 0x10], %o4
	ldx		[%o1 + 0x18], %o5
	ldx		[%o1 + 0x20], %g1
	stxa		%o2, [%o0] ASI_BLK_INIT_QUAD_LDD_P
	stxa		%o2, [%o0] ASI_ST_BLKINIT_MRU_P
	add		%o0, 0x08, %o0
	ldx		[%o1 + 0x28], %g2
	stxa		%o3, [%o0] ASI_BLK_INIT_QUAD_LDD_P
	stxa		%o3, [%o0] ASI_ST_BLKINIT_MRU_P
	add		%o0, 0x08, %o0
	ldx		[%o1 + 0x30], %g3
	stxa		%o4, [%o0] ASI_BLK_INIT_QUAD_LDD_P
	stxa		%o4, [%o0] ASI_ST_BLKINIT_MRU_P
	add		%o0, 0x08, %o0
	ldx		[%o1 + 0x38], %o2
	add		%o1, 0x40, %o1
	stxa		%o5, [%o0] ASI_BLK_INIT_QUAD_LDD_P
	stxa		%o5, [%o0] ASI_ST_BLKINIT_MRU_P
	add		%o0, 0x08, %o0
	stxa		%g1, [%o0] ASI_BLK_INIT_QUAD_LDD_P
	stxa		%g1, [%o0] ASI_ST_BLKINIT_MRU_P
	add		%o0, 0x08, %o0
	stxa		%g2, [%o0] ASI_BLK_INIT_QUAD_LDD_P
	stxa		%g2, [%o0] ASI_ST_BLKINIT_MRU_P
	add		%o0, 0x08, %o0
	stxa		%g3, [%o0] ASI_BLK_INIT_QUAD_LDD_P
	stxa		%g3, [%o0] ASI_ST_BLKINIT_MRU_P
	add		%o0, 0x08, %o0
	stxa		%o2, [%o0] ASI_BLK_INIT_QUAD_LDD_P
	stxa		%o2, [%o0] ASI_ST_BLKINIT_MRU_P
	add		%o0, 0x08, %o0
	bne,pt		%icc, 1b
	 prefetch	[%o1 + 0x200], #n_reads_strong
Loading