原理圖
LED 原理圖如下圖所示,連接GPIO19,GPIO21,GPIO22。

image
按鍵 BUTTON 原理圖如下圖所示,連接 BTN1 和 BTN2 。

image
物理連接如下圖所示。
開發(fā)板的按鍵引腳 BTN1 和 BTN2 分別連接到 GPIO30 和 GPIO31 。

image
源碼
開發(fā)板的按鍵引腳連接到 GPIO30 和 GPIO31 。

image
主程序如下:
#include <stdio.h>
#include <stdlib.h>
#include "platform.h"
#include <string.h>
#include "plic/plic_driver.h"
#include "encoding.h"
#include <unistd.h>
#include "stdatomic.h"
// Structures for registering different interrupt handlers
// for different parts of the application.
typedef void (*function_ptr_t) (void);
void no_interrupt_handler (void) {};
function_ptr_t g_ext_interrupt_handlers[PLIC_NUM_INTERRUPTS];
// Instance data for the PLIC.
plic_instance_t g_plic;
//處理外部中斷
void handle_m_ext_interrupt(){
plic_source int_num = PLIC_claim_interrupt(&g_plic);
if ((int_num >=1 ) && (int_num < PLIC_NUM_INTERRUPTS)) {
g_ext_interrupt_handlers[int_num]();
}
else {
exit(1 + (uintptr_t) int_num);
}
PLIC_complete_interrupt(&g_plic, int_num);
}
void button_1_handler(void) {
printf("open led");
// Green LED On
GPIO_REG(GPIO_OUTPUT_VAL) |= (1 << RED_LED_GPIO_OFFSET);
GPIO_REG(GPIO_OUTPUT_VAL) |= (1 << GREEN_LED_GPIO_OFFSET);
GPIO_REG(GPIO_OUTPUT_VAL) |= (1 << BLUE_LED_GPIO_OFFSET);
// Clear the GPIO Pending interrupt by writing 1.
GPIO_REG(GPIO_RISE_IP) = (0x1 << BUTTON_1_GPIO_OFFSET);
};
void button_2_handler(void) {
printf("close led");
GPIO_REG(GPIO_OUTPUT_VAL) &= ~(1 << RED_LED_GPIO_OFFSET);
GPIO_REG(GPIO_OUTPUT_VAL) &= ~(1 << GREEN_LED_GPIO_OFFSET);
GPIO_REG(GPIO_OUTPUT_VAL) &= ~(1 << BLUE_LED_GPIO_OFFSET);
GPIO_REG(GPIO_RISE_IP) = (0x1 << BUTTON_2_GPIO_OFFSET);
};
void register_plic_irqs (){
/**************************************************************************
* Set up the PLIC
*
*************************************************************************/
PLIC_init(&g_plic,
PLIC_CTRL_ADDR,
PLIC_NUM_INTERRUPTS,
PLIC_NUM_PRIORITIES);
for (int ii = 0; ii < PLIC_NUM_INTERRUPTS; ii ++){
g_ext_interrupt_handlers[ii] = no_interrupt_handler;
}
g_ext_interrupt_handlers[PLIC_INT_DEVICE_BUTTON_1] = button_1_handler;
g_ext_interrupt_handlers[PLIC_INT_DEVICE_BUTTON_2] = button_2_handler;
// Have to enable the interrupt both at the GPIO level,
// and at the PLIC level.
PLIC_enable_interrupt (&g_plic, PLIC_INT_DEVICE_BUTTON_1);
PLIC_enable_interrupt (&g_plic, PLIC_INT_DEVICE_BUTTON_2);
// Priority must be set > 0 to trigger the interrupt.
PLIC_set_priority(&g_plic, PLIC_INT_DEVICE_BUTTON_1, 1);
PLIC_set_priority(&g_plic, PLIC_INT_DEVICE_BUTTON_2, 1);
}
int main(int argc, char **argv)
{
//關(guān)閉LED GPIO的輸入使能
GPIO_REG(GPIO_INPUT_EN) &= ~((0x1<< RED_LED_GPIO_OFFSET) | (0x1<< GREEN_LED_GPIO_OFFSET) | (0x1 << BLUE_LED_GPIO_OFFSET)) ;
//打開LED GPIO的輸出使能:三個顏色的LED燈
GPIO_REG(GPIO_OUTPUT_EN) |= ((0x1<< RED_LED_GPIO_OFFSET)| (0x1<< GREEN_LED_GPIO_OFFSET) | (0x1 << BLUE_LED_GPIO_OFFSET)) ;
//設(shè)置LED GPIO的輸出值
GPIO_REG(GPIO_OUTPUT_VAL) &= ~ (0x1 << RED_LED_GPIO_OFFSET) ;
GPIO_REG(GPIO_OUTPUT_VAL) &= ~ (0x1 << GREEN_LED_GPIO_OFFSET) ;
GPIO_REG(GPIO_OUTPUT_VAL) &= ~ (0x1 << BLUE_LED_GPIO_OFFSET) ;
//關(guān)閉按鍵GPIO的輸出使能
GPIO_REG(GPIO_OUTPUT_EN) &= ~((0x1<< BUTTON_1_GPIO_OFFSET) | (0x1<< BUTTON_2_GPIO_OFFSET)) ;
//關(guān)閉按鍵GPIO的輸入上拉
GPIO_REG(GPIO_PULLUP_EN) &= ~((0x1<< BUTTON_1_GPIO_OFFSET) | (0x1<< BUTTON_2_GPIO_OFFSET)) ;
//打開GPIO的輸入使能
GPIO_REG(GPIO_INPUT_EN) |= ((0x1<< BUTTON_1_GPIO_OFFSET) | (0x1<< BUTTON_2_GPIO_OFFSET)) ;
//上升沿觸發(fā)的中斷來源
GPIO_REG(GPIO_RISE_IE) |= ((0x1<< BUTTON_1_GPIO_OFFSET) | (0x1<< BUTTON_2_GPIO_OFFSET)) ;
printf("set gpio finish");
//關(guān)閉外部中斷局部使能
clear_csr(mie, MIP_MEIP);
//計時器初始化
register_plic_irqs();
// 打開外部中斷局部使能
set_csr(mie, MIP_MEIP);
//打開中斷的全局使能
set_csr(mstatus, MSTATUS_MIE);
printf("set interrupt gpio finish");
return 0;
}
實驗結(jié)果
按鍵 BTN1 會將 LED 燈關(guān)閉,按鍵 BTN2 會將 LED 燈打開。