C語言操作寄存器和函數(shù)指針
typedef函數(shù)指針類型
#include <stdio.h>
//函數(shù)指針類型別名
/*
* int 函數(shù)返回值
* (int,int)函數(shù)參數(shù),兩個參數(shù)int,int
* *PTP_TO_FUNC函數(shù)指針,指向函數(shù)的指針
*/
typedef int (*PTR_TO_FUNC)(int, int);
/*
為數(shù)組定義別名與函數(shù)指針類型別名類似
[4]數(shù)組各屬
PTR_TO_ARR指向數(shù)組的名,其數(shù)組個數(shù)與參數(shù)個數(shù)相同
在使用是當(dāng)成一種類型,在為其賦值時需要重新為其添加值
*/
typedef char(*PTR_TO_ARR)[10];
//實現(xiàn)函數(shù)體
int max(int x, int y)
{
return x > y ? x : y;
}
int main(void)
{
//定義數(shù)組,等待指向
char str[3][10] = {
"嘿嘿",
"信息科技",
"有限公司"
};
// PTR_TO_ARR結(jié)構(gòu)體指針,仍需要定義別名
PTR_TO_ARR arr = str[1];
// 指向函數(shù)
PTR_TO_FUNC func = max;
printf("max(6,3): %d\n", (*func)(6, 3));
printf("str[1]: %s\n", (*arr)); //輸出信息科技
return0;
}
寄存器
有限存貯容量的高速存貯部件 。寄存器的功能是存儲二進(jìn)制代碼,它是由具有存儲功能的觸發(fā)器組合起來構(gòu)成的。一個觸發(fā)器可以存儲1位二進(jìn)制代碼,故存放n位二進(jìn)制代碼的寄存器,需要n個觸發(fā)器來構(gòu)成。
寄存器分類
基本寄存器 :只能并行送入數(shù)據(jù),也只能并行輸出。
移位寄存器中的數(shù)據(jù)可以在移位脈沖作用下依次逐位右移或左移,數(shù)據(jù)既可以并行輸入,并行輸出,也可以串行輸入,串行輸出,還可以并行輸入,串行輸出或串行輸入,并行輸出,靈活,用途廣泛。
使用
在嵌入式編程中,常常需要對一些寄存器進(jìn)行配置,有的情況下需要改變一1個字節(jié)中的某一位或者幾位,但是又不想改變其它位原有的值,就可以使用按位運(yùn)算符進(jìn)行操作。
假如我們只需要設(shè)置第0位bit0的值為1時, 要保持其它位 不發(fā)生變化。
TEST = 0x01
此方式如果高7位沒有使用,就不會有影響,但是如果高7位正在被使用,那么就會發(fā)生錯誤。
- 與運(yùn)算 :對于二進(jìn)制位操作,不管原值是0還是1,它跟0進(jìn)行&與運(yùn)算,得到的結(jié)果都是 0,而和1進(jìn)行&運(yùn)算,將保持原來的值不變
- 或運(yùn)算 :不管該位原來的值是0還是1,它跟1進(jìn)行 |運(yùn)算,得到的結(jié)果都是1,而跟0運(yùn)算,將保持原來的值不變。`
可以使用或運(yùn)算:
TEST = TEST | 0x01;
// 在實際中常用
TEST |= 0x01;
給Test的低4位清 0 ,高四位保持不變:
TEST &= 0xF0; //使用十六進(jìn)制
此方法在單片機(jī)中經(jīng)常使用,先對需要設(shè)置的位用 &操作符進(jìn)行清零操作,然后用 | 操作符設(shè)置值,改變GPIOA的狀態(tài),先對寄存器的值進(jìn)行清零操作,然后根據(jù)需要設(shè)置的值進(jìn)行 | 或運(yùn)算:
GPIOA->CRL &= 0XFFFFFF0F; // 將第4~7位清零
GPIOA->CRL &= 0X00000040; //設(shè)置相應(yīng)的值,不改變其他位的值
移位提高可讀性
GPIOx->BSRR = (((uint32_t)0x01) << pinpox); //將0x01 左移pinpox位,
通過左移而不是直接設(shè)置一個固定的值 :為了提高代碼的可讀性,直接就知道修改了第幾位:
GPIOA->ODR |= 1<<5; //PA.5輸出高,其它位不變
設(shè)置某位為0
簡單操作:
TIMx->SR = 0xFFF7; //此方法仍然影響可讀性,
庫函數(shù):
TIMx -> SR = (uint16_t)~TIM_FLAG;
TIM_FLAG定義
設(shè)置SR的第三位為 0 時即可設(shè)置為
TIMx->SR = (uint16_t)~TIM_FLAG_CC3;
#define TIM_FLAG_Update ((uint16_t)0x0001)
#define TIM_FLAG_CC1 ((uint16_t)0x0002)
#define TIM_FLAG_CC2 ((uint16_t)0x0004)
#define TIM_FLAG_CC3 ((uint16_t)0x0008)
#define TIM_FLAG_CC4 ((int16_t)0x0010)
#define TIM_FLAG_COM ((uint16_t)0x0020)
#define TIM_FLAG_Trigger ((uint16_t)0x0040)
#define TIM_FLAG_Break ((uint16_t)0x0080)
#define TIM_FLAG_CC1OF ((uint16_t)0x0200)
#define TIM_FLAG_CC2OF ((uint16_t)0x0400)
#define TIM_FLAG_CC3OF ((uint16_t)0x0800)
#define TIM_FLAG_CC4OF ((uint16_t)0x1000)
位域
位域:或稱之為位段,英文表達(dá)式 Bit field 是一種數(shù)據(jù)結(jié)構(gòu),可以把數(shù)據(jù)以位元的形式緊湊的存儲,并允許程序員對此結(jié)構(gòu)進(jìn)行位元進(jìn)行操作。
優(yōu)勢:
- 可以使數(shù)據(jù)單元節(jié)省存儲空間
- 位段可以很方便地訪問一個整數(shù)值的部分內(nèi)容從而簡化程序源代碼。
位域可以分為兩大類,一個是結(jié)構(gòu)體位域,一個是共同體位域,由于共同體和結(jié)構(gòu)體兩者在定義上的形式都是相同的,從位域的定義形式上看,兩者也基本都是相同的。
struct 位域結(jié)構(gòu)體
{
類型說明符 位域名 : 長度;
}結(jié)構(gòu)體變量名;
// 結(jié)構(gòu)體位域
struct example0
{
unsignedchar x : 3; //冒號后面的證書指定了該位段所占用的位的數(shù)目。
unsignedchar y : 2;
unsignedchar z : 1;
}ex0_t;
// 共同體位域
union example1
{
unsignedchar x : 3;
unsignedchar y : 2;
unsignedchar z : 1;
}ex1_u;
位域大小原則 :整個結(jié)構(gòu)體位域的總大小為最寬基本類型成員大小的整數(shù)倍。
位域基本都使用無符號類型。
位域注意
- 結(jié)構(gòu)體位域成員不能使用取址操作
- 結(jié)構(gòu)體成員不能夠使用static修飾
- 結(jié)構(gòu)體位域成員不能使用數(shù)組。
不同的處理器,不同的編譯器對位域的影響,位域雖然能夠以位的形式操作數(shù)據(jù),但是也被人們告知要慎重使用,原因在于不同的處理器結(jié)構(gòu),不同的編譯器對于位域的一些特征會產(chǎn)生不同的結(jié)果。
處理器大端模式,小端模式的處理器也會對下面的結(jié)構(gòu)體位域產(chǎn)生不一樣的存儲方式。
不同的編譯器,結(jié)構(gòu)體位域成員不同類型,不同的編譯器對于位域會有不同的結(jié)果
當(dāng)成員大小之和超過一個基本存儲空間時,不同的編譯器也會有不同的處理方式。
typedefunion
{
unsignedchar Byte;
struct
{
unsignedchar bit012 : 3;
unsignedchar bit34 : 2;
unsignedchar bit5 : 1;
unsignedchar bit6 : 1;
unsignedchar bit7 : 1;
}bits;
}registerType;
存儲0x0000 8000定義一個指針指向地址:
registerType *pReg = (register*)0x00008000;
// 使用位域寄存器進(jìn)行賦值
pReg->bits.bit5 = 1;
pReg->bits.bit012 = 7;