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

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

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

C語言開發(fā)中可能會用到的GNU

Q4MP_gh_c472c21 ? 來源:開源博客 ? 作者:-_-struggle ? 2021-11-17 10:41 ? 次閱讀

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

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

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

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

char data[0];

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

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

case 范圍

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

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

語句表達(dá)式


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

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

這種寫法可以避免:

 #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ù)處理器去掉它前面的那個逗號。如果你在宏調(diào)用時,確實提供了一些可變參數(shù),GNU C也會工作正常,它會把這些可變參數(shù)放到逗號的后面。

標(biāo)號元素

標(biāo)準(zhǔn)C要求數(shù)組或結(jié)構(gòu)體的初始化值必須以固定的順序出現(xiàn),在GNU C中,通過指定索引或結(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ù)定義兩個標(biāo)志符保存當(dāng)前函數(shù)的名字,__ FUNCTION __ 保存函數(shù)在源碼中的名字, __ PRETTY__ FUNCTION __保存帶語言特色的名字。在C函數(shù)中這兩個名字是相同的。

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

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

特殊屬性聲明

GNU C 允許聲明函數(shù)、變量和類型的特殊屬性,以便進(jìn)行手工的代碼優(yōu)化和定制。如果要指定一個屬性聲明,只需要在聲明后添加__ attribute __((ATTRIBUTE))。其中ATTRIBUTE為屬性說明,如果存在多個屬性,則以逗號分隔。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等十個屬性。

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

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

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

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

regparm屬性用于指定最多可以使用n個寄存器(eax, edx, ecx)傳遞參數(shù),n的范圍是0~3,超過n時則將參數(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屬性里使用的寄存器個數(shù)超過3個,也仍然會使用其他寄存器來傳遞參數(shù)。這一點要遵循X64體系結(jié)構(gòu)的調(diào)用約定。

下面可以看一個例子:

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選項用于反匯編所有的程序段,包括:代碼段、數(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ī)定完全不一致了。所以,在這里作者覺得,regparm已經(jīng)不起作用了。

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

    關(guān)注

    180

    文章

    7581

    瀏覽量

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

    關(guān)注

    0

    文章

    143

    瀏覽量

    17434

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

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

收藏 人收藏

    評論

    相關(guān)推薦

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

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

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

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

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

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

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

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

    C語言C++那些不同的地方

    ++11標(biāo)準(zhǔn)。根據(jù)不同的標(biāo)準(zhǔn),它們的功能也會有所不同,但是越新的版本支持的編譯器越少,所以本文在討論的時候使用的C語言標(biāo)準(zhǔn)是C89,C++標(biāo)準(zhǔn)是C
    的頭像 發(fā)表于 12-07 14:29 ?850次閱讀
    <b class='flag-5'>C</b><b class='flag-5'>語言</b>和<b class='flag-5'>C</b>++<b class='flag-5'>中</b>那些不同的地方

    常用的c語言開發(fā)環(huán)境有哪些

    C語言是一種廣泛應(yīng)用于系統(tǒng)編程、嵌入式開發(fā)和科學(xué)計算等領(lǐng)域的高級編程語言。為了能夠高效地開發(fā)C
    的頭像 發(fā)表于 11-27 16:14 ?5298次閱讀

    C語言運行環(huán)境是什么

    C語言運行環(huán)境(C language runtime environment)是指在執(zhí)行C語言程序時所需的軟件及硬件環(huán)境。
    的頭像 發(fā)表于 11-27 16:13 ?3142次閱讀

    嵌入式C語言的結(jié)構(gòu)特點

    嵌入式開發(fā)既有底層硬件的開發(fā)又涉及上層應(yīng)用的開發(fā),即涉及系統(tǒng)的硬件和軟件,C語言既具有匯編
    的頭像 發(fā)表于 11-24 16:16 ?584次閱讀
    嵌入式<b class='flag-5'>C</b><b class='flag-5'>語言</b>的結(jié)構(gòu)特點

    \0在c語言中怎么用

    是由貝爾實驗室的Dennis Ritchie為了開發(fā)UNIX操作系統(tǒng)而設(shè)計的。它在20世紀(jì)70年代初首次出現(xiàn),并迅速流行起來。C語言以其簡潔、高效、低級別的特性成為了廣泛使用的編程語言
    的頭像 發(fā)表于 11-24 09:59 ?2885次閱讀

    C語言開發(fā)DSP系統(tǒng)全過程

    電子發(fā)燒友網(wǎng)站提供《用C語言開發(fā)DSP系統(tǒng)全過程.pdf》資料免費下載
    發(fā)表于 11-18 10:57 ?0次下載
    用<b class='flag-5'>C</b><b class='flag-5'>語言</b><b class='flag-5'>開發(fā)</b>DSP系統(tǒng)全過程

    C語言開發(fā)DSP系統(tǒng)的全過程講解

    電子發(fā)燒友網(wǎng)站提供《用C語言開發(fā)DSP系統(tǒng)的全過程講解.pdf》資料免費下載
    發(fā)表于 11-18 10:53 ?0次下載
    用<b class='flag-5'>C</b><b class='flag-5'>語言</b><b class='flag-5'>開發(fā)</b>DSP系統(tǒng)的全過程講解

    c語言嵌入式開發(fā)

    電子發(fā)燒友網(wǎng)站提供《c語言嵌入式開發(fā).zip》資料免費下載
    發(fā)表于 11-17 14:11 ?2次下載
    <b class='flag-5'>c</b><b class='flag-5'>語言</b>嵌入式<b class='flag-5'>開發(fā)</b>

    C語言中鏈表的作用是什么?

    C語言中指針用的很少,鏈表、文件操作幾乎沒用過,所以也不能理解到底有什么作用。各位有經(jīng)常在做程序時會用到這些嗎。
    發(fā)表于 11-06 06:23

    開發(fā)ARM單片機(jī)時會不會用到JSON?

    開發(fā)ARM單片機(jī)時,會不會用到JSON 會在哪些方面用到?如何使用?
    發(fā)表于 10-30 06:32

    如何用C語言開發(fā)DSP嵌入式系統(tǒng)?

    如何用C語言開發(fā)DSP嵌入式系統(tǒng)? DSP嵌入式系統(tǒng)開發(fā)是一個非常重要的領(lǐng)域,它涉及到設(shè)備的控制、數(shù)據(jù)的處理、系統(tǒng)的優(yōu)化等多種方面,因此,掌握這方面的技術(shù)非常重要。 在本文中,將詳細(xì)探
    的頭像 發(fā)表于 10-29 17:29 ?1109次閱讀