;------------------------------------------------------------------------------ ; ; 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: ; ; CopyMem.asm ; ; Abstract: ; ; CopyMem function ; ; Notes: ; ;------------------------------------------------------------------------------ .686 .model flat,C .xmm .code ;------------------------------------------------------------------------------ ; VOID * ; InternalMemCopyMem ( ; IN VOID *Destination, ; IN VOID *Source, ; IN UINTN Count ; ); ;------------------------------------------------------------------------------ InternalMemCopyMem PROC USES esi edi mov esi, [esp + 16] ; esi <- Source mov edi, [esp + 12] ; edi <- Destination mov edx, [esp + 20] ; edx <- Count lea eax, [esi + edx - 1] ; eax <- End of Source cmp esi, edi jae @F cmp eax, edi ; Overlapped? jae @CopyBackward ; Copy backward if overlapped @@: xor ecx, ecx sub ecx, edi and ecx, 15 ; ecx + edi aligns on 16-byte boundary jz @F cmp ecx, edx cmova ecx, edx sub edx, ecx ; edx <- remaining bytes to copy rep movsb @@: mov ecx, edx and edx, 15 shr ecx, 4 ; ecx <- # of DQwords to copy jz @CopyBytes add esp, -16 movdqu [esp], xmm0 ; save xmm0 @@: movdqu xmm0, [esi] ; esi may not be 16-bytes aligned movntdq [edi], xmm0 ; edi should be 16-bytes aligned add esi, 16 add edi, 16 loop @B mfence movdqu xmm0, [esp] ; restore xmm0 add esp, 16 ; stack cleanup jmp @CopyBytes @CopyBackward: mov esi, eax ; esi <- Last byte in Source lea edi, [edi + edx - 1] ; edi <- Last byte in Destination std @CopyBytes: mov ecx, edx rep movsb cld mov eax, [esp + 12] ; eax <- Destination as return value ret InternalMemCopyMem ENDP END