C语言extern引用AT&T汇编中的变量,任意转换类型

今天研究出了一个小问题,在C语言里引用汇编的变量,会是什么结果,汇编中的变量没有像C语言中int类型那样的类型约束,可以把数据当作任何类型处理,那么传到C语言中我们应该当作什么类型处理呢。

换句话说,在汇编里定义变量var,在C语言里引用,我们肯定要用extern声明var外部变量,那么extern后面接什么类型?难道是extern int var么?还是 extern short var

实例:

注意:混合编译的方法,gcc test.s main.c 这样直接用gcc处理汇编和C语言文件就可以自动搞定。

一、

汇编程序:(是的,就这么短小)

.global var #注意现在的汇编器不再要求被C语言引用的变量名前加下划线
.data
var:
.fill 10,4,9 #填充10个单元,每个单元4字节,每个单元的值为9

C语言:

#include <stdio.h>
extern int var; //现在的C语言编译器引用汇编变量可以使用和汇编变量同样的名字,写成_var反而会错误
void main()
{
printf("sizeof(var)=%d\n",sizeof(var));
printf("%d\n", var);
}

程序的输出为

sizeof(int)=4
9

解释很简单,




    		    C语言extern引用AT&T汇编中的变量,任意转换类型

如上图,我只画了四个字节,我填充了10个单元的,每单元占用4字节,意思就是每个9占四个字节。如图,然后C语言里声明为extern int var;就是把前四个字节看作一个int变量取出来,当然是9了。

二、

保持C语言程序和上一个一样,

extern int var;

把汇编语句改成

.fill 10,2,9 #10个单元,每个单元填入9,每单元大小为2B

这就意味着每个单元2字节,每个单元填一个9,我们不改变C语言语句,那么仍将把var看作int型4字节变量,依然是一次取出前4个字节作为var的值。那我们推测一下结果会是什么。

首先每个9占用两字节,内存图应该是这样的,(只画了4字节,后面没画)




    		    C语言extern引用AT&T汇编中的变量,任意转换类型

一次性取出4个字节,就是0x00090009,这玩意儿拿到windows自带计算器里算出来是是十进制的589833,这就是我们的预测值。让我们看下程序结果是不是和我们猜的一样。




    		    C语言extern引用AT&T汇编中的变量,任意转换类型

没错吧,这就是说我们在汇编里定义的变量,来到C语言里可以任意声明为任意类型,从sizeof的输出也能看出,我们把var声明成多大,它就是多大。

三、

我们再试试别的,汇编语句保持和上一个一样,依然是

.fill 10,2,9 #10个单元,每个单元填入9,每单元大小为2B

我们改变C语言语句,把var声明成

extern long long var; //8字节整型

当然,输出语句稍微变下,因为是long long型整数,所以要这么输出printf("%lld\n", var);

#include <stdio.h>
extern long long var;
void main()
{
printf("sizeof(var)=%d\n",sizeof(var));
printf("%lld\n", var);
}

我们还是事先猜测结果,字节太多我不画图了,按照上面的的思路,这次取出8字节当作long long变量var的值,那么应该是0x0009000900090009,化为十进制是2533313445691401。

看看程序运行结果:




    		    C语言extern引用AT&T汇编中的变量,任意转换类型

预测正确。

四、

再来,把var声明为char变量。

#include <stdio.h>
extern char var; #声明为char
void main()
{
printf("sizeof(var)=%d\n",sizeof(var));
printf("%c\n", var);
}

输出语句相应的改变成%c

汇编里面稍微改改填充的数值,因为9是制表符的ASCII码,输出了你也看不到。我们改成填充43

.fill 10,2,43 #10个单元,每个单元填入43(内存实际存储按照二进制来的),每单元大小为2B

注意43的十六进制表示为0x2b,这是加号“+”的ASCII码,

内存里面长成这样




    		    C语言extern引用AT&T汇编中的变量,任意转换类型

C语言会认为var是char型的,取出一个字节当作var的值,也就是把0x2b送给var,我们把var用%c输出,就是输出了0x2b这个ASCII码所代表的字符。就是“+”号。




    		    C语言extern引用AT&T汇编中的变量,任意转换类型

五、

好吧,再这么整下去也没完了,下面用一个数组定义来终结此文吧。

汇编语言代码和上一个一样,

.fill 10,2,43

C语言修改成:

#include <stdio.h>
typedef char CHARARR[20]; //CHARARR是一个类型别名,用它定义的每一个变量都是包含20个字符的数组
extern CHARARR var; //var被定义为CHARARR类型,是一个包含20个元素的char型数组。
void main()
{
printf("sizeof(var)=%d\n",sizeof(var));
int i=0;
while(i<=19)
{
printf("[%c]", var[i]); //用[]括起来更容易看出输出现象
i++;
}
printf("\n");
}

我们让var成为数组,含有20个元素,正好把汇编里面的 10单元x2字节 共计20个字节包括了。

然后用循环逐个输出,为什么要这么做而不用%s呢,因为数组里有很多字节是0x00,这是字符串终止标志。%s会被打断的。

结果如下:




    		    C语言extern引用AT&T汇编中的变量,任意转换类型

可以看到和预计的一样,20个元素以ASCII码 0x2b 0x00 0x2b 0x00。。。。这样的数组顺序挨个输出。

讲完了,C语言和汇编语言结合起来是不是强大到了难以想象的地步,随心所欲操纵内存数据。

本文出自 “Strive For RedHat Inc.” 博客,请务必保留此出处http://mirage1993.blog.51cto.com/2709744/1635436

更多相关文章
  •   1. 下载libmad源码   2. 在目录C:\cygwin\home\Administrator\android-ndk-r4b-windows\android-ndk-r4b\samples\下创建目录jni,并把libmad的源码解压到该目录下,这样mad.h的目录是:C:\cygwin ...
  • 据位于伦敦的安全公司ScanSafe发表的数据显示,今年第2季度微软的IE浏览器中的一个安全漏洞是被黑客用来实施攻击的最多的一个安全漏洞. 这家安全公司称,排名第一的安全漏洞是"Exploit.HTML.Mht".黑客利用微软IE浏览器中的这个安全漏洞就可以在用户计算机上下载和安 ...
  • Given an unsorted array of integers, find the length of the longest consecutive elements sequence. For example,Given [100, 4, 200, 1, 3, 2],The longes ...
  • 一.远程桌面授权 远程桌面授权(RD 授权)以前称为终端服务授权(TS 授权),它能够管理每个设备或用户与远程桌面会话主机(RD 会话主机)服务器连接所需的远程桌面服务客户端访问许可 (RDS CAL).使用 RD 授权在远程桌面授权服务器上安装.颁发 RDS CAL 并跟踪其可用性. 客户端(用户 ...
  • 背景详见<html" target=_blank>Hosts绑定新思路之DNS代理篇>核心内容1. DNS协议解析2. 启动UDP服务,监听53端口3. 根据DB或者文本,进行Hosts解析DNS协议DNS Protocol Overview (推荐)非强详细,但是不怎么 ...
  • 最近几乎每天都有应急响应工作,自己写的一个linux下的webshell查杀小脚本,如果网站文件太多,打包太大,可以用这个脚本查杀一下,然后根据日志和时间再找其他的webshell,如果网站文件比较小,建议打包下来,用 {D盾Web后门查杀V1.2.6版本} 在windows下查杀,查杀脚本下载地址 ...
一周排行
  • 最近客户的多个页面中了远程加载一些代码的页面,主要是利用MSXML2.serverXMLHTTP来加载一些代码并执行,下面是具体的解决方法. 解决办法: 1.用青云团队开发的网站木马清理专家全面扫描服务器上的网站,网 ...
  • 简单说一下本人对次时代框架的理解. 首先对次时代框架的感觉有两点:很强大且日志记录很详细.是对struts框架的重写. 1.在站点的最前端应该放有一个apache服务器,来进行负载平衡和实现其它一些简单的功能,如记录 ...
  • 1:以 root为例 [[email protected] deploy]# ssh-keygen  直接回车到底   通过ssh-cpoy-id自带脚本来拷贝公钥到客户机 命令格式: [[email protected] ~]# ssh-co ...
  • 本文给出NAT类型判断方法. Cone NAT or Symmetric NAT 具体实现框图如下:若SERVER A和SERVER B返回给Client的PORT1和PORT2相等,则该NAT为Cone NAT,否 ...
  • 本视频是Linux系统工程师集训之网络部分视频课程的第十五集. 主要内容为访问控制列表. 工作中发现很多搞Linux系统运维的人由于缺乏基础的网络知识而遇到瓶颈,而且还是一个比较常见的现象(拜国内不实用的教育系统所赐 ...
  • CNET科技资讯网3月31日国际报道 BeyondTrust公布的最新报告显示,Windows 7使用者权限若设定为一般使用者,而不是管理员身分,则可平息90%的Windows 7重大安全威胁.BeyondTrust ...
  • 联想某重要分站POST注入漏洞 sqlmap -u "http://yun.lenovo.com/join.php" --data="keywords=-1&search=&qu ...
  •   from T_COMPANY c join T_COMPANY_POSITION p on c.ID = p.COMPANYID order by :type desc nulls last; 最初不知道不能这样 ...
  • mysql的dump导库命令   最近用到mysql的dump导库表,将该命令的使用方法记录下来:    www.2cto.com   mysqldump备份: mysqldump -u用户名 -p密码 -h主机 数 ...
  • jQuery想必大部分前端er都知道甚至很熟悉了,网上有数以万计的优秀的jQuery插件以及教程,今天收集了一些关于图片.相册的jQuery插件代码,希望会对你有所帮助. 1. 3D Gallery 2. Circu ...