文章目錄

  • 前言
  • 一、SDK包簡介
  • 二、SDK文件使用
  • 1.文件移植
  • 2.宏定義數據類型
  • 3.編寫主程序
  • 三、下載編譯
  • 總結

前言




提示:以下是本篇文章正文內容

一、SDK包簡介

上一節中,外設要用的寄存器都是我們自己定義,其實,在SDK包裏已經編寫好了外設寄存器的定義,我們可以直接移植用。

NXP 針對 I.MX6ULL 編寫了一個 SDK 包,SDK 包就類似於 STM32 的 STD 庫或者HAL 庫,SDK 包提供了 Windows 和 Linux 兩種版本,分別針對主機系統是 Windows 和Linux

我們只需要 關注SDK 包裏面與寄存器定義相關的文件,一共需要如下三個文件:
(1)fsl_common.h:SDK_2.2_MCIM6ULL\devices\MCIMX6Y2\drivers\fsl_common.h

(2)fsl_iomuxc.h: SDK_2.2_MCIM6ULL\devices\MCIMX6Y2\drivers\fsl_iomuxc.h。

(3)MCIMX6Y2.h: SDK_2.2_MCIM6ULL\devices\MCIMX6Y2\MCIMX6YH2.h。

設備為MCIMX6Y2

二、SDK文件使用

1.文件移植

使用VSCode 新建工程,將 fsl_common.h、 fsl_iomuxc.h 和 MCIMX6Y2.h 這三個文件拷貝到工程中,需要對其做刪減,偷個懶我直接用的正點原子已經移植好的三個文件😁

2.宏定義數據類型

創建一個.h文件名字可以隨便取(cc.h),主要用來宏定義SDK文件所需用到的數據類型

#ifndef __CC_H
#define __CC_H

#define     __I     volatile 
#define     __O     volatile 
#define     __IO    volatile

typedef   signed          char int8_t;
typedef   signed short     int int16_t;
typedef   signed           int int32_t;
typedef unsigned          char uint8_t;
typedef unsigned short     int uint16_t;
typedef unsigned           int uint32_t;
typedef unsigned long     long uint64_t;
typedef	  signed char  	 	   s8;		
typedef	  signed short 	  int  s16;
typedef	  signed int 		   s32;
typedef	  signed long long int s64;
typedef	unsigned char 		   u8;
typedef	unsigned short int     u16;
typedef	unsigned int 		   u32;
typedef	unsigned long long int u64;


#endif

3.編寫主程序

(1)IOMUXC_SetPinMux()
該函數在在文件 fsl_iomuxc.h 定義
功能:設 置 IO 復 用 功 能

函數原型

static inline void IOMUXC_SetPinMux(uint32_t muxRegister,
									uint32_t muxMode,
									uint32_t inputRegister,
									uint32_t inputDaisy,
									uint32_t configRegister,
									uint32_t inputOnfield)
{
	*((volatile uint32_t *)muxRegister) =
	IOMUXC_SW_MUX_CTL_PAD_MUX_MODE(muxMode) |
	IOMUXC_SW_MUX_CTL_PAD_SION(inputOnfield);
	if (inputRegister)
	{
		*((volatile uint32_t *)inputRegister) =
		IOMUXC_SELECT_INPUT_DAISY(inputDaisy);
	}
}

muxRegister:IO 的 復 用 寄 存 器 地 址
muxMode:IO 複用值, ALT0~ALT8,對應數字 0~8
inputRegister: 外設輸入 IO 選擇寄存器地址
inputDaisy: 寄存器 inputRegister 的值
configRegister:未使用,IOMUXC_SetPinConfig 會使用這個寄存器
inputOnfield : IO軟件輸使能

使用此函數將 GPIO1_IO03 的複用功能設置為GPIO

IOMUXC_SetPinMux(IOMUXC_GPIO1_IO03_GPIO1_IO03, 0);

為什麼只傳入兩個參數呢?注意,這裏的IOMUXC_GPIO1_IO03_GPIO1_IO03是一個宏定義

#define IOMUXC_GPIO1_IO03_GPIO1_IO03     0x020E0068U, 0x5U, 
0x00000000U, 0x0U, 0x020E02F4U

將這個宏替換進去函數就變成了

IOMUXC_SetPinMux (0x020E0068U, 0x5U, 0x00000000U, 0x0U, 0x020E02F4U, 0);

剛好是6個參數,與之一 一對應

NXP 的 SDK 庫將一個 IO 的所有複用功能都定義了一個宏,比如GPIO1_IO3有9個宏定義

IOMUXC_GPIO1_IO03_I2C1_SDA
IOMUXC_GPIO1_IO03_GPT1_COMPARE3
IOMUXC_GPIO1_IO03_USB_OTG2_OC
IOMUXC_GPIO1_IO03_USDHC1_CD_B
IOMUXC_GPIO1_IO03_GPIO1_IO03
IOMUXC_GPIO1_IO03_CCM_DI0_EXT_CLK
IOMUXC_GPIO1_IO03_SRC_TESTER_ACK
IOMUXC_GPIO1_IO03_UART1_RX
IOMUXC_GPIO1_IO03_UART1_TX

(2)IOMUXC_SetPinConfig()
在是在文件 fsl_iomuxc.h 中定義的
功能:配置IO的屬性,如IO的上下拉,速度

函數原型

static inline void IOMUXC_SetPinConfig(	uint32_t muxRegister,
										uint32_t muxMode,
										uint32_t inputRegister,
										uint32_t inputDaisy,
										uint32_t configRegister,
										uint32_t configValue)
{
	if (configRegister)
	{
		*((volatile uint32_t *)configRegister) = configValue;
	}
}

muxRegister:IO 的 復 用 寄 存 器 地 址
muxMode:IO 複用值, ALT0~ALT8,對應數字 0~8
inputRegister: 外設輸入 IO 選擇寄存器地址
inputDaisy: 寄存器 inputRegister 的值
configRegister:未使用,IOMUXC_SetPinConfig 會使用這個寄存器
configValue: 寫入到寄存器 configRegister 的值

配置IO屬性

IOMUXC_SetPinConfig(IOMUXC_GPIO1_IO03_GPIO1_IO03, 0X10B0);

宏替換後

IOMUXC_SetPinConfig(0x020E0068U, 0x5U, 0x00000000U, 0x0U, 0x020E02F4U, 0X10B0);

這兩個函數聯合起來配置IO的複用功能和IO配置

以前配置IO

IOMUX_SW_MUX->GPIO1_IO03 = 0X5; /* 複用為 GPIO1_IO03 */
IOMUX_SW_PAD->GPIO1_IO03 = 0X10B0; /* IO屬性*/

現在

/* 1、初始化 IO 複用 */
IOMUXC_SetPinMux(IOMUXC_GPIO1_IO03_GPIO1_IO03,0);
IOMUXC_SetPinConfig(IOMUXC_GPIO1_IO03_GPIO1_IO03,0X10B0);

和前面的程序基本説一樣,就差不多改了這兩句

三、下載編譯

可以直接使用前面的Makefile文件和鏈接腳本imx6ul.lds
這裏對Makefile文件修改了以下使用了變量簡化編寫過程

CROSS_COMPILE ?= arm-linux-gnueabihf-
NAME ?= ledc

CC := $(CROSS_COMPILE)gcc
LD := $(CROSS_COMPILE)ld
OBJCOPY := $(CROSS_COMPILE)objcopy
OBJDUMP := $(CROSS_COMPILE)objdump

OBJS := start.o main.o

$(NAME).bin:$(OBJS)
	$(LD) -Timx6ul.lds -o $(NAME).elf $^
	$(OBJCOPY) -O binary -S $(NAME).elf $@
	$(OBJDUMP) -D -m arm $(NAME).elf > $(NAME).dis

%.o:%.s
	$(CC) -Wall -nostdlib -c -O2 -o $@ $<

%.o:%.S
	$(CC) -Wall -nostdlib -c -O2 -o $@ $<

%.o:%.c
	$(CC) -Wall -nostdlib -c -O2 -o $@ $<

clean:
	rm -rf *.o $(NAME).bin $(NAME).elf $(NAME).dis

小記:Makefile各種等號的作用
(1)賦值符 =
使用“=”在給變量的賦值的時候,不一定要用已經定義好的值,也可以使用後面定義的值,變量的值為最後一次賦值的值

eg:

name = ming
curname = $(name)
name = mingfei

print:
	@echo curname: $(curname)

則在終端輸出:mingfei即最後一次賦值的值

(2)賦值符 :=
不會使用後面定義的變量,只能使用前面定義好的變量

name = ming
curname := $(name)
name = mingfei

print:
	@echo curname: $(curname)

則在終端會輸出:ming

(3)賦值符 ?=
如果該變量前面沒有被賦值,則該變量等於 ?= 後面的值,反之,則使用前面的賦過的值

name = ming
curname ?= $(name)
name = mingfei

print:
	@echo curname: $(curname)

則終端會輸出:ming

(4)變量追加 +=
在變量後面追加一些字符串(在Makefile中,變量是字符串類型)

objs = mian.o start.o
objs += led.o

則在終端會輸出:mian.o start.o led.o


總結

提示:這裏對文章進行總結: