memcopy()函数c语言实现和汇编实现比较

今天突然心血来潮想起我前一家公司面试时遇到的一个面试题,就是实现个memcopy()函数。当初太紧张(刚毕业第二次面试),所以写的不是很好(可以说漏洞百出);现在刚学了点汇编,刚好就用两种语言实现下;

首先来看汇编实现的memcpy函数,是利用宏函数来实现的,用汇编指令rep和movsb配合循环把数据以字节为单位从ds:esi传送到es:edi中,把循环次数放在ecx中。当然这样拷贝的才是真正的内存拷贝,其他的函数都稍微有点牵强。

 #define myMemcpy(dest, src, n) ({         void* _ret = dest;         __asm__("cld;rep;movsb"             ::"D"((long)(_ret)), "S"((long)(src)), "c"((long)(n)));         _ret;})

在调试汇编实现的memcpy函数时,遇到个错误,这里记录下来:

错误:myMemcpy.c: In function ‘main’:

myMemcpy.c:15: error: can‘t find a register in class ‘CREG’ while reloading ‘asm’

myMemcpy.c:15: error: ‘asm’ operand has impossible constraints

发生这个错误是因为,我开始添加了 修改寄存器列表那一栏(添加了变更寄存器si, di, ex);查了下资料说是同一个变量两次出现时使用了同一个寄存器,就会报这样的错误。解决办法是把变更寄存器中那些寄存器名称用占位符替代(di==0;si==1;cx==2);但是还是不行,后来我干脆就把  修改寄存器列表 删了,因为我记得有些gcc编译器对于在输入输出中出现过的寄存器,不列入变更寄存器列表中;(我前几篇blog中也提到在输入输出中出现过的寄存器,不用添加到变更寄存器列表中去)但是有些版本的gcc却需要添加到变更寄存器那一栏;

下面来看下c语言方式来实现memcpy函数,其实我感觉c语言来实现还是比较简单的,就是把输入的数据强制转换成字符来操作,因为在c语言中没有比字符更小的单位了。而对于是否返回ret,如果要多次使用memcpy的话(   myMemcpy1(xxx,  myMemcpy1(xx, xx, xx),  xx)    ),那肯定是要返回的。如果不需要就没必要和原来的memcpy()一样,不要太死板(不过还是建议要返回ret,可以考虑以后扩展)。

 void* myMemcpy1(void* dest, void* src, unsigned int n)
 {
     char* d = (char*)dest;
     char* s = (char*)src;
     char* ret = d;

     while(n--){
         *d++ = *s++;
     }
     return ret;
 }

好了,该实现的都实现了,现在来测试下,当然我调试通过了,感兴趣的可以试着调试下。

 int main(void)
 {
     char test1[] = "yuzhihui";
     char test2[1024] = {};
     char test3[1024] = {}; 

     myMemcpy(test2, test1, sizeof(test1));
     myMemcpy1(test3, test1, sizeof(test1));

     printf("test1:%s\n", test1);
     printf("test2:%s\n", test2);
     printf("test3:%s\n", test3);

     int t1 = 100;
     int t2 = 0;
     int t3 = 0;

     myMemcpy(&t2, &t1, sizeof(t1));
     myMemcpy1(&t3, &t1, sizeof(t1));

     printf("t1:%d\n", t1);
     printf("t2:%d\n", t2);
     printf("t3:%d\n", t3);

     return 0;
 }

下面就贴下运行的结果:

转载请注明作者和原文出处,原文地址:http://blog.csdn.net/yuzhihui_no1/article/details/43601757

如果有什么不正确之处,欢迎大家指正,一起努力,共同学习!!

时间: 2024-10-30 02:53:39

memcopy()函数c语言实现和汇编实现比较的相关文章

[Linux内核分析第一周课程] 由C语言程序的汇编表示观察CPU寄存器与内存的互动

孟宁<Linux内核分析>第一周实验 作者:Zou Le 原创作品转载请注明出处. 课程信息: <Linux内核分析>MOOC课程http://mooc.study.163.com/course/USTC-1000029000 ---------------------------实验正文--------------------------- 本实验在实验楼64位LIinux虚拟机下进行. C代码如下: int increment5(int x) { return x + 5; }

嵌入式Linux ARM汇编(七)——C语言与ARM汇编混合编程

嵌入式Linux ARM汇编(七)--C语言与ARM汇编混合编程 在嵌入式系统开发中,目前使用的主要编程语言是C和汇编.在大规模的嵌入式软件中,例如含有OS,大部分的代码都是用C编写的,主要是因为C语言的结构比较好,便于人的理解,而且有大量的支持库.但是很多地方还是要用到汇编语言,例如开机时硬件系统的初始化,包括CPU状态的设定,中断的使能,主频的设定,以及RAM的控制参数及初始化,一些中断处理方面也可能涉及汇编.另外一个使用汇编的地方就是一些对性能非常敏感的代码块,这是不能依靠C编译器的生成代

Linux下用C语言调用GAS汇编——综合实例

紧接上一篇博文,上篇写的是用GAS汇编调用C语言,这次讲用C语言调用GAS汇编.在内核编写中主要用的是前者,但是在日常程序优化中,主要用的是后者. 还是同样的功能,实现两个数值的交换,把参数从C语言传递进汇编,再用C语言接收从汇编函数返回的返回值,返回值又分为普通整数返回值和字符串返回值. 建立三个文件,分别为main.c.retstr.s.swapint.s.其中main.c是C语言文件,也就是主函数所在,swapint.s是汇编函数,用于交换两个变量数值并返回一个整数值给C语言函数,rets

[汇编] C语言中嵌入汇编

>_<" 下面是在C语言中嵌入汇编的例子,下面是三点要注意的~ 1.内联式汇编 2._asm关键字 3.并不是所有中断都能被支持 1 #include<iostream> 2 #include<windows.h> 3 #include<tchar.h> 4 using namespace std; 5 6 int _tmain(int argc, _TCHAR * argv[]) 7 { 8 9 int data1=0x000000; 10 in

在c语言中嵌入汇编语句,对于我来说相当难。

今天早上在csdn论坛上看到一个帖子http://topic.csdn.net/u/20120917/14/82f42e17-977a-4824-95bd-7b79db15d283.html:“C语言中嵌入汇编,究竟有何意义?” 其中看到一个例子是在c语言中插入一段汇编代码获取CPU的主频,制造商和型号的: //=====================================================================================/*      

Ok6410裸机驱动学习(三)C语言内嵌汇编

1.C语言内嵌汇编使用方法 C内嵌汇编以关键字”_asm_或asm开始,下辖4个部分,各部分之间用“:”分开,第一部分是必须写的,后面3个部分可以省略,但是分号:不能省略 优化后的代码 2.汇编程序框架 .section .data <初始化的数据> .section .bss <未初始化的数据> .section .text .global _start _start: <汇编代码>

C语言内嵌汇编

1.C语言内嵌汇编 1.1内嵌汇编的语法 1.2内嵌汇编示例 #include <stdio.h> int main() { int result = 0; int input = 1; int a = 1; int b = 2; asm volatile ( "movl %1, %0\n" // 通过占位符指定交互的变量 : "=r"(result) // 输出变量,与汇编交互 : "r"(input) // 输出变量,与汇编交互

3203 数组做函数参数----排序函数--C语言版

3203: 数组做函数参数----排序函数--C语言版 时间限制: 1 Sec  内存限制: 128 MB提交: 253  解决: 151[提交][状态][讨论版][命题人:smallgyy] 题目描述 定义一个函数来完成对参数数组中元素的排序工作,函数声明如下: void sort(int array[ ]);//该函数完成对array数组的排序工作 在以下程序的基础上,完成对sort函数的定义,提交时,只提交sort函数的定义即可. #include <stdio.h> void sort

3204: 数组做函数参数--排序函数2--C语言

3204: 数组做函数参数--排序函数2--C语言 时间限制: 1 Sec  内存限制: 128 MB提交: 211  解决: 143[提交][状态][讨论版][命题人:smallgyy] 题目描述 定义一个函数来完成对参数数组中元素的排序工作,函数声明如下: void sort(int array[ ],int n);//该函数完成对array数组的前n个元素的排序工作 在以下程序的基础上,完成对sort函数的定义,提交时,只提交sort函数的定义即可. #include <stdio.h >