RM新时代网站-首页

0
  • 聊天消息
  • 系統(tǒng)消息
  • 評(píng)論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫(xiě)文章/發(fā)帖/加入社區(qū)
會(huì)員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識(shí)你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

C語(yǔ)言開(kāi)發(fā)中可能會(huì)用到的GNU

Q4MP_gh_c472c21 ? 來(lái)源:開(kāi)源博客 ? 作者:-_-struggle ? 2021-11-17 10:41 ? 次閱讀

為了方便使用,GNU C在標(biāo)準(zhǔn)C語(yǔ)言的基礎(chǔ)上進(jìn)行了部分方便開(kāi)發(fā)的擴(kuò)展。

這里講解一些開(kāi)發(fā)中可能會(huì)用到的,或者使用頻率比較高的內(nèi)容。

零長(zhǎng)度數(shù)組和變量長(zhǎng)度數(shù)組

GNU C 允許使用零長(zhǎng)度數(shù)組,比如:

char data[0];

GNU C 允許使用一個(gè)變量定義數(shù)組的長(zhǎng)度如:

int n=0;
scanf("%d",&n);
int array[n];

case 范圍

GNU C支持 case x...y這樣的語(yǔ)法,[x,y]之間數(shù)均滿足條件。

case 'a'...'z':  /*from 'a' to 'z'*/
break;

語(yǔ)句表達(dá)式


GNU C 把包含在括號(hào)中的復(fù)合語(yǔ)句看作是一個(gè)表達(dá)式,稱為語(yǔ)句表達(dá)式。

 #define min_t(type,x,y)
         ({type __x=(x); type __y=(y);__x<__y?__x:__y;})

這種寫(xiě)法可以避免:

 #define min_t(x,y) ((x)<(y)?(x):(y))

在min_t(x++,++y)中出現(xiàn)的副作用。

typeof 關(guān)鍵字

typeof(x)可以獲得x的類型借助typeof關(guān)鍵字我們可以重新定義min_t:

#define min_t(x,y)
    ({typeof(x) __x=(x); typeof(y) __y=(y);__x<__y?__x:__y;})

可變參數(shù)宏

GNU C中宏也支持可變參數(shù):

#define pr_debug(fmt,arg...) 
        printk(fmt,##arg)

這里,如果可變參數(shù)被忽略或?yàn)榭?,?#”操作將使預(yù)處理器去掉它前面的那個(gè)逗號(hào)。如果你在宏調(diào)用時(shí),確實(shí)提供了一些可變參數(shù),GNU C也會(huì)工作正常,它會(huì)把這些可變參數(shù)放到逗號(hào)的后面。

標(biāo)號(hào)元素

標(biāo)準(zhǔn)C要求數(shù)組或結(jié)構(gòu)體的初始化值必須以固定的順序出現(xiàn),在GNU C中,通過(guò)指定索引或結(jié)構(gòu)體成員名,允許初始化以任意順序出現(xiàn)。

unsigned char data[MAX] =
{
         [0]=10,
         [10]=100,
};


struct file_operations ext2_file_operations=
{
        open:ext2_open,
        close:ext2_close,
};

linux 2.6中推薦如下方式:

struct file_operations ext2_file_operations=
{
     .read=ext2_read,
     .write=ext2_write,
};

當(dāng)前函數(shù)名

GNU C中預(yù)定義兩個(gè)標(biāo)志符保存當(dāng)前函數(shù)的名字,__ FUNCTION __ 保存函數(shù)在源碼中的名字, __ PRETTY__ FUNCTION __保存帶語(yǔ)言特色的名字。在C函數(shù)中這兩個(gè)名字是相同的。

void func_example()
{
     printf("the function name is %s",__FUNCTION__);
}

在C99中支持__ func __ 宏,因此建議使用 __ func __ 替代 __ FUNCTION __ 。

特殊屬性聲明

GNU C 允許聲明函數(shù)、變量和類型的特殊屬性,以便進(jìn)行手工的代碼優(yōu)化和定制。如果要指定一個(gè)屬性聲明,只需要在聲明后添加__ attribute __((ATTRIBUTE))。其中ATTRIBUTE為屬性說(shuō)明,如果存在多個(gè)屬性,則以逗號(hào)分隔。GNU C 支持noreturn,noinline, always_inline, pure, const, nothrow, format, format_arg, no_instrument_function, section, constructor, destructor, used, unused, deprecated, weak, malloc, alias warn_unused_result nonnull等十個(gè)屬性。

noreturn屬性作用于函數(shù),表示該函數(shù)從不返回。這會(huì)讓編譯器優(yōu)化代碼并消除不必要的警告信息。例如:

#define ATTRIB_NORET __attribute__((noreturn)) ....
asmlinkage NORET_TYPE void do_exit(long error_code) ATTRIB_NORET;

packed屬性作用于變量和類型,用于變量或結(jié)構(gòu)域時(shí),表示使用最小可能的對(duì)齊,用于枚舉、結(jié)構(gòu)或聯(lián)合類型時(shí)表示該類型使用最小的內(nèi)存。如對(duì)于結(jié)構(gòu)體,就是它告訴編譯器取消結(jié)構(gòu)在編譯過(guò)程中的優(yōu)化對(duì)齊,按照實(shí)際占用字節(jié)數(shù)進(jìn)行對(duì)齊。例如:

struct example_struct
{
         char a;
         int b;
         long c;
} __attribute__((packed));

regparm屬性用于指定最多可以使用n個(gè)寄存器(eax, edx, ecx)傳遞參數(shù),n的范圍是0~3,超過(guò)n時(shí)則將參數(shù)壓入棧中(n=0表示不用寄存器傳遞參數(shù))。

注意:以上這些屬性都是在X86處理器體系結(jié)構(gòu)下的,在X64體系結(jié)構(gòu)下,大部分內(nèi)容都是同樣有效的。但是,這里要注意regparm屬性,由于在X64體系結(jié)構(gòu)下,GUN C的默認(rèn)調(diào)用約定使用寄存器傳遞參數(shù)。所以,如果regparm屬性里使用的寄存器個(gè)數(shù)超過(guò)3個(gè),也仍然會(huì)使用其他寄存器來(lái)傳遞參數(shù)。這一點(diǎn)要遵循X64體系結(jié)構(gòu)的調(diào)用約定。

下面可以看一個(gè)例子:

int q = 0x5a;
int t1 = 1;
int t2 = 2;
int t3 = 3;
int t4 = 4;
#define REGPARM3 __attribute((regparm(3)))
#define REGPARM0 __attribute((regparm(0)))
void REGPARM0 p1(int a)
{
     q = a + 1;
}


void REGPARM3 p2(int a, int b, int c, int d)
{
     q = a + b + c + d + 1;
}


int main()
{
    p1(t1);
    p2(t1,t2,t3,t4);
    return 0;
}

使用objdump命令反匯編,相關(guān)命令如下:

objdump -D 可執(zhí)行程序

其中-D選項(xiàng)用于反匯編所有的程序段,包括:代碼段、數(shù)據(jù)段、只讀數(shù)據(jù)段以及一些系統(tǒng)段等等。而-d命令只反匯編代碼段的內(nèi)容。

反匯編后的關(guān)鍵代碼如下:

Disassembly of section .text:
0000000000400474 :
  400474:    55                       push   %rbp
  400475:    48 89 e5                 mov    %rsp,%rbp
  400478:    89 7d fc                 mov    %edi,-0x4(%rbp)
  40047b:    8b 45 fc                 mov    -0x4(%rbp),%eax
  40047e:    83 c0 01                 add    $0x1,%eax
  400481:    89 05 3d 04 20 00        mov    %eax,0x20043d(%rip)        # 6008c4 
  400487:    c9                       leaveq 
  400488:    c3                       retq   


0000000000400489 :
  400489:    55                       push   %rbp
  40048a:    48 89 e5                 mov    %rsp,%rbp
  40048d:    89 7d fc                 mov    %edi,-0x4(%rbp)
  400490:    89 75 f8                 mov    %esi,-0x8(%rbp)
  400493:    89 55 f4                 mov    %edx,-0xc(%rbp)
  400496:    89 4d f0                 mov    %ecx,-0x10(%rbp)
  400499:    8b 45 f8                 mov    -0x8(%rbp),%eax
  40049c:    8b 55 fc                 mov    -0x4(%rbp),%edx
  40049f:    8d 04 02                 lea    (%rdx,%rax,1),%eax
  4004a2:    03 45 f4                 add    -0xc(%rbp),%eax
  4004a5:    03 45 f0                 add    -0x10(%rbp),%eax
  4004a8:    83 c0 01                 add    $0x1,%eax
  4004ab:    89 05 13 04 20 00        mov    %eax,0x200413(%rip)        # 6008c4 
  4004b1:    c9                       leaveq 
  4004b2:    c3                       retq   


00000000004004b3 
: 4004b3: 55 push %rbp 4004b4: 48 89 e5 mov %rsp,%rbp 4004b7: 53 push %rbx 4004b8: 8b 05 0a 04 20 00 mov 0x20040a(%rip),%eax # 6008c8 4004be: 89 c7 mov %eax,%edi 4004c0: e8 af ff ff ff callq 400474 4004c5: 8b 0d 09 04 20 00 mov 0x200409(%rip),%ecx # 6008d4 4004cb: 8b 15 ff 03 20 00 mov 0x2003ff(%rip),%edx # 6008d0 4004d1: 8b 1d f5 03 20 00 mov 0x2003f5(%rip),%ebx # 6008cc 4004d7: 8b 05 eb 03 20 00 mov 0x2003eb(%rip),%eax # 6008c8 4004dd: 89 de mov %ebx,%esi 4004df: 89 c7 mov %eax,%edi 4004e1: e8 a3 ff ff ff callq 400489 4004e6: b8 00 00 00 00 mov $0x0,%eax 4004eb: 5b pop %rbx 4004ec: c9 leaveq 4004ed: c3 retq 4004ee: 90 nop 4004ef: 90 nop Disassembly of section .data: 00000000006008c0 <__data_start>: 6008c0: 00 00 add %al,(%rax) ... 00000000006008c4 : 6008c4: 5a pop %rdx 6008c5: 00 00 add %al,(%rax) ... 00000000006008c8 : 6008c8: 01 00 add %eax,(%rax) ... 00000000006008cc : 6008cc: 02 00 add (%rax),%al ... 00000000006008d0 : 6008d0: 03 00 add (%rax),%eax ... 00000000006008d4 : 6008d4: 04 00 add $0x0,%al ...

如果讀者還記得2.2.3節(jié)中,關(guān)于GCC基于X64體系結(jié)構(gòu)的調(diào)用約定的話,那就很容易可以看出,函數(shù)p1和p2都使用寄存器傳遞參數(shù),順序就是RDI, RSI, RDX, RCX,這些細(xì)節(jié)已經(jīng)跟regparm的規(guī)定完全不一致了。所以,在這里作者覺(jué)得,regparm已經(jīng)不起作用了。

來(lái)源:https://my.oschina.net/LinuxDaxingxing/blog/751319
責(zé)任編輯:haq
聲明:本文內(nèi)容及配圖由入駐作者撰寫(xiě)或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場(chǎng)。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問(wèn)題,請(qǐng)聯(lián)系本站處理。 舉報(bào)投訴
  • C語(yǔ)言
    +關(guān)注

    關(guān)注

    180

    文章

    7604

    瀏覽量

    136683
  • GNU
    GNU
    +關(guān)注

    關(guān)注

    0

    文章

    143

    瀏覽量

    17492

原文標(biāo)題:你知道GNU C對(duì)C語(yǔ)言的擴(kuò)展嗎?

文章出處:【微信號(hào):gh_c472c2199c88,微信公眾號(hào):嵌入式微處理器】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。

收藏 人收藏

    評(píng)論

    相關(guān)推薦

    語(yǔ)言模型開(kāi)發(fā)語(yǔ)言是什么

    在人工智能領(lǐng)域,大語(yǔ)言模型(Large Language Models, LLMs)背后,離不開(kāi)高效的開(kāi)發(fā)語(yǔ)言和工具的支持。下面,AI部落小編為您介紹大語(yǔ)言模型
    的頭像 發(fā)表于 12-04 11:44 ?94次閱讀

    MCU編程語(yǔ)言開(kāi)發(fā)環(huán)境介紹

    微控制器單元(Microcontroller Unit,簡(jiǎn)稱MCU)是嵌入式系統(tǒng)的核心,廣泛應(yīng)用于各種電子產(chǎn)品。隨著技術(shù)的發(fā)展,MCU編程語(yǔ)言開(kāi)發(fā)環(huán)境也在不斷進(jìn)步,以適應(yīng)不同的應(yīng)用需求。 1.
    的頭像 發(fā)表于 11-01 11:51 ?660次閱讀

    C語(yǔ)言C++結(jié)構(gòu)體的區(qū)別

    同樣是結(jié)構(gòu)體,看看在C語(yǔ)言C++中有什么區(qū)別?
    的頭像 發(fā)表于 10-30 15:11 ?197次閱讀

    GNU構(gòu)建裸機(jī)系統(tǒng)

    基于AT91SAM7S平臺(tái),介紹裸機(jī)開(kāi)發(fā),以閃燈為藍(lán)本,涉及匯編、鏈接、C/C++、中斷等。   無(wú)處不在的ARM處理器家族得到了GNU C
    發(fā)表于 10-16 17:34 ?0次下載

    hex文件如何查看原c語(yǔ)言代碼

    直接將 .hex 文件轉(zhuǎn)換回原始的 C 語(yǔ)言代碼是不可能的,因?yàn)?.hex 文件是二進(jìn)制文件,它包含了單片機(jī)程序編譯后的機(jī)器碼,這些機(jī)器碼與原始的 C
    的頭像 發(fā)表于 09-02 10:37 ?2122次閱讀

    請(qǐng)問(wèn)ESP-AT在編譯過(guò)程中會(huì)用到哪些源文件?

    如題,請(qǐng)問(wèn)ESP-AT在編譯過(guò)程中會(huì)用到哪些源文件?要修改藍(lán)牙相關(guān)的代碼應(yīng)該修該哪些呢?有通透的大神指點(diǎn)一下嗎?謝謝!
    發(fā)表于 06-27 06:59

    上位機(jī)軟件開(kāi)發(fā)用什么語(yǔ)言

    維護(hù)的上位機(jī)軟件至關(guān)重要。本文將詳細(xì)介紹幾種常用的上位機(jī)軟件開(kāi)發(fā)編程語(yǔ)言,并分析它們的優(yōu)缺點(diǎn)。 C/C++ C
    的頭像 發(fā)表于 06-06 10:44 ?1751次閱讀

    嵌入式C語(yǔ)言結(jié)構(gòu)體基本實(shí)現(xiàn)

    C語(yǔ)言中的數(shù)組只能允許程序員定義存儲(chǔ)相同類型數(shù)據(jù)。但是結(jié)構(gòu)是C語(yǔ)言編程中允許您存儲(chǔ)不同數(shù)據(jù)類型的數(shù)據(jù)。 結(jié)構(gòu)體的定義 ????要想定義結(jié)構(gòu),必須用到
    的頭像 發(fā)表于 05-11 08:49 ?1011次閱讀
    嵌入式<b class='flag-5'>中</b><b class='flag-5'>C</b><b class='flag-5'>語(yǔ)言</b>結(jié)構(gòu)體基本實(shí)現(xiàn)

    C語(yǔ)言:嵌入式開(kāi)發(fā)的關(guān)鍵編譯器角色

    嵌入式程序開(kāi)發(fā)跟硬件密切相關(guān),需要使用C語(yǔ)言來(lái)讀寫(xiě)底層寄存器、存取數(shù)據(jù)、控制硬件等,C語(yǔ)言和硬件之間由編譯器來(lái)聯(lián)系,一些
    發(fā)表于 04-26 14:53 ?610次閱讀
    <b class='flag-5'>C</b><b class='flag-5'>語(yǔ)言</b>:嵌入式<b class='flag-5'>開(kāi)發(fā)</b><b class='flag-5'>中</b>的關(guān)鍵編譯器角色

    幫你避雷 C語(yǔ)言所謂的短路現(xiàn)象~

    C語(yǔ)言短路現(xiàn)象算是C語(yǔ)言的基礎(chǔ)吧,不過(guò)有時(shí)候代碼寫(xiě)得不規(guī)范也容易引入一些bug,所以這些操作在工程師實(shí)踐盡量少用。雖然下面找的例子比較簡(jiǎn)單
    的頭像 發(fā)表于 03-27 08:09 ?410次閱讀
    幫你避雷 <b class='flag-5'>C</b><b class='flag-5'>語(yǔ)言</b>所謂的短路現(xiàn)象~

    plc編程語(yǔ)言c語(yǔ)言的聯(lián)系 c語(yǔ)言和PLC有什么區(qū)別

    PLC編程語(yǔ)言C語(yǔ)言的聯(lián)系 PLC(可編程邏輯控制器)是一種針對(duì)自動(dòng)化控制系統(tǒng)的特殊計(jì)算機(jī)。PLC編程語(yǔ)言是為了控制和管理自動(dòng)化生產(chǎn)過(guò)程
    的頭像 發(fā)表于 02-05 14:21 ?4089次閱讀

    c語(yǔ)言,c++,java,python區(qū)別

    C語(yǔ)言C++、Java和Python是四種常見(jiàn)的編程語(yǔ)言,各有優(yōu)點(diǎn)和特點(diǎn)。 C語(yǔ)言
    的頭像 發(fā)表于 02-05 14:11 ?2360次閱讀

    vb語(yǔ)言c++語(yǔ)言的區(qū)別

    Microsoft開(kāi)發(fā)的一種面向?qū)ο蟮氖录?qū)動(dòng)編程語(yǔ)言。它的設(shè)計(jì)目標(biāo)是簡(jiǎn)化編程過(guò)程,讓初學(xué)者也能快速上手。與之相比,C++語(yǔ)言是一種通用的、面向?qū)ο蟮木幊?/div>
    的頭像 發(fā)表于 02-01 10:20 ?2252次閱讀

    求助,如何將C++代碼從GNU移植到Tasking編譯器?

    需要解決的障礙。 我懇請(qǐng)你幫助我們解決這個(gè)問(wèn)題。 問(wèn)題 :當(dāng)前代碼是使用 GNU 編譯器編譯的,代碼按定義運(yùn)行。 但是,我們希望讓它在 Aurix TC399 開(kāi)發(fā)套件上運(yùn)行。 在此過(guò)程,我
    發(fā)表于 01-31 07:29

    嵌入式開(kāi)發(fā)常見(jiàn)的C語(yǔ)言技巧與方法分享

    在嵌入式開(kāi)發(fā),常常要操作寄存器,對(duì)寄存器進(jìn)行寫(xiě)入,讀出等等操作。每個(gè)寄存器都有自己固有的地址,通過(guò)C語(yǔ)言訪問(wèn)這些地址就變得尤為重要。
    的頭像 發(fā)表于 12-26 09:55 ?1094次閱讀
    RM新时代网站-首页