audk/EdkCompatibilityPkg/Foundation/Library/EfiCommonLib/Ia32/EfiSetMem.S

159 lines
3.6 KiB
ArmAsm

#/*++
#
#Copyright (c) 2006, Intel Corporation. All rights reserved.<BR>
#This program and the accompanying materials
#are licensed and made available under the terms and conditions of the BSD License
#which accompanies this distribution. The full text of the license may be found at
#http://opensource.org/licenses/bsd-license.php
#
#THE PROGRAM IS DISTRIBUTED UNDER THE BSD LICENSE ON AN "AS IS" BASIS,
#WITHOUT WARRANTIES OR REPRESENTATIONS OF ANY KIND, EITHER EXPRESS OR IMPLIED.
#
#Module Name:
#
# EfiSetMem.asm
#
#Abstract:
#
# This is the code that supports IA32-optimized SetMem service
#
#--*/
#include "EfiBind.h"
#---------------------------------------------------------------------------
.686:
#.MODEL flat,C
.mmx:
.code:
#---------------------------------------------------------------------------
.globl ASM_PFX(EfiCommonLibSetMem)
#VOID
#EfiCommonLibSetMem (
# IN VOID *Buffer,
# IN UINTN Count,
# IN UINT8 Value
# )
#/*++
#
#Input: VOID *Buffer - Pointer to buffer to write
# UINTN Count - Number of bytes to write
# UINT8 Value - Value to write
#
#Output: None.
#
#Saves:
#
#Modifies:
#
#Description: This function is an optimized set-memory function.
#
#Notes: This function tries to set memory 8 bytes at a time. As a result,
# it first picks up any misaligned bytes, then words, before getting
# in the main loop that does the 8-byte clears.
#
#--*/
ASM_PFX(EfiCommonLibSetMem):
pushl %ebp
movl %esp, %ebp
subl $0x10, %esp # Reserve space for local variable UINT64 QWordValue @[ebp - 10H] & UINT64 MmxSave @[ebp - 18H]
pushl %ebx
pushl %edi
movl 0xC(%ebp), %edx # Count
testl %edx, %edx
je _SetMemDone
pushl %ebx
movl 8(%ebp), %eax # Buffer
movb 0x10(%ebp), %bl # Value
movl %eax, %edi
movb %bl, %bh
cmpl $256, %edx
jb _SetRemindingByte
andb $0x7, %al
testb %al, %al
je _SetBlock
movl %edi, %eax
shrl $3, %eax
incl %eax
shll $3, %eax
subl %edi, %eax
cmpl %edx, %eax
jnb _SetRemindingByte
subl %eax, %edx
movl %eax, %ecx
movb %bl, %al
rep
stosb
_SetBlock:
movl %edx, %eax
shrl $6, %eax
testl %eax, %eax
je _SetRemindingByte
shll $6, %eax
subl %eax, %edx
shrl $6, %eax
movw %bx, -0x10(%ebp) # QWordValue[0]
movw %bx, -0x10+2(%ebp) # QWordValue[2]
movw %bx, -0x10+4(%ebp) # QWordValue[4]
movw %bx, -0x10+6(%ebp) # QWordValue[6]
movq %mm0, -8(%ebp) # Save mm0 to MmxSave
movq -0x10(%ebp), %mm0 # Load QWordValue to mm0
_B:
movq %mm0, %ds:(%edi)
movq %mm0, %ds:8(%edi)
movq %mm0, %ds:16(%edi)
movq %mm0, %ds:24(%edi)
movq %mm0, %ds:32(%edi)
movq %mm0, %ds:40(%edi)
movq %mm0, %ds:48(%edi)
movq %mm0, %ds:56(%edi)
addl $64, %edi
decl %eax
jnz _B
# Restore mm0
movq -8(%ebp), %mm0 # Restore MmxSave to mm0
emms # Exit MMX Instruction
_SetRemindingByte:
movl %edx, %ecx
movl %ebx, %eax
shll $16, %eax
movw %bx, %ax
shrl $2, %ecx
rep
stosl
movl %edx, %ecx
andl $3, %ecx
rep
stosb
popl %ebx
_SetMemDone:
popl %edi
popl %ebx
leave
ret
#EfiCommonLibSetMem ENDP