AP_HAL_Linux: Fix RPI check using enumerations

This commit is contained in:
Mohammad Hefny 2023-06-01 15:38:09 +03:00 committed by Randy Mackay
parent bde600a05d
commit 2f61448eec
5 changed files with 88 additions and 57 deletions

View File

@ -160,14 +160,14 @@ void GPIO_RPI::closeMemoryDevice()
void GPIO_RPI::init() void GPIO_RPI::init()
{ {
const int rpi_version = UtilRPI::from(hal.util)->get_rpi_version(); const LINUX_BOARD_TYPE rpi_version = UtilRPI::from(hal.util)->detect_linux_board_type();
GPIO_RPI::Address peripheral_base; GPIO_RPI::Address peripheral_base;
if(rpi_version == 0) { if(rpi_version == LINUX_BOARD_TYPE::RPI_ZERO_1) {
peripheral_base = Address::BCM2708_PERIPHERAL_BASE; peripheral_base = Address::BCM2708_PERIPHERAL_BASE;
} else if (rpi_version == 1 || rpi_version == 2) { } else if (rpi_version == LINUX_BOARD_TYPE::RPI_2_3_ZERO2) {
peripheral_base = Address::BCM2709_PERIPHERAL_BASE; peripheral_base = Address::BCM2709_PERIPHERAL_BASE;
} else if (rpi_version == 3) { } else if (rpi_version == LINUX_BOARD_TYPE::RPI_4) {
peripheral_base = Address::BCM2711_PERIPHERAL_BASE; peripheral_base = Address::BCM2711_PERIPHERAL_BASE;
} else { } else {
AP_HAL::panic("Unknown rpi_version, cannot locate peripheral base address"); AP_HAL::panic("Unknown rpi_version, cannot locate peripheral base address");

View File

@ -24,7 +24,7 @@
#include "GPIO.h" #include "GPIO.h"
#include "RCInput_RPI.h" #include "RCInput_RPI.h"
#include "Util_RPI.h"
#ifdef DEBUG #ifdef DEBUG
#define debug(fmt, args ...) do { fprintf(stderr,"[RCInput_RPI]: %s:%d: " fmt, __FUNCTION__, __LINE__, ## args); } while (0) #define debug(fmt, args ...) do { fprintf(stderr,"[RCInput_RPI]: %s:%d: " fmt, __FUNCTION__, __LINE__, ## args); } while (0)
@ -152,12 +152,12 @@ Memory_table::Memory_table()
} }
// Init Memory table // Init Memory table
Memory_table::Memory_table(uint32_t page_count, int version) Memory_table::Memory_table(const uint32_t page_count, const LINUX_BOARD_TYPE version)
{ {
uint32_t i; uint32_t i;
int fdMem, file; int fdMem, file;
// Cache coherent adresses depends on RPI's version // Cache coherent adresses depends on RPI's version
uint32_t bus = version == 0 ? 0x40000000 : 0xC0000000; uint32_t bus = version == LINUX_BOARD_TYPE::RPI_ZERO_1 ? 0x40000000 : 0xC0000000;
uint64_t pageInfo; uint64_t pageInfo;
void *offset; void *offset;
@ -199,7 +199,7 @@ Memory_table::Memory_table(uint32_t page_count, int version)
munmap(_virt_pages[i], PAGE_SIZE); munmap(_virt_pages[i], PAGE_SIZE);
void * prev_virt = _virt_pages[i]; void * prev_virt = _virt_pages[i];
_virt_pages[i] = mmap(_virt_pages[i], PAGE_SIZE, PROT_READ | PROT_WRITE, MAP_SHARED | MAP_FIXED | MAP_NORESERVE | MAP_LOCKED, fdMem, ((uintptr_t)_phys_pages[i] & (version == 0 ? 0xFFFFFFFFFFFFFFFF : ~bus))); _virt_pages[i] = mmap(_virt_pages[i], PAGE_SIZE, PROT_READ | PROT_WRITE, MAP_SHARED | MAP_FIXED | MAP_NORESERVE | MAP_LOCKED, fdMem, ((uintptr_t)_phys_pages[i] & (version == LINUX_BOARD_TYPE::RPI_ZERO_1 ? 0xFFFFFFFFFFFFFFFF : ~bus)));
if (_virt_pages[i] == MAP_FAILED) { if (_virt_pages[i] == MAP_FAILED) {
fprintf(stderr, "Failed phys2virt prev_virt=%p phys_page=%p %m\n", prev_virt, _phys_pages[i]); fprintf(stderr, "Failed phys2virt prev_virt=%p phys_page=%p %m\n", prev_virt, _phys_pages[i]);
printf("Make sure that CONFIG_STRICT_DEVMEM is disabled\n"); printf("Make sure that CONFIG_STRICT_DEVMEM is disabled\n");
@ -274,22 +274,22 @@ uint32_t Memory_table::get_page_count() const
// Physical addresses of peripheral depends on Raspberry Pi's version // Physical addresses of peripheral depends on Raspberry Pi's version
void RCInput_RPI::set_physical_addresses() void RCInput_RPI::set_physical_addresses()
{ {
if (_version == 0) { if (_version == LINUX_BOARD_TYPE::RPI_ZERO_1) {
// 1 & zero are the same // 1 & zero are the same
dma_base = RCIN_RPI_RPI1_DMA_BASE; dma_base = RCIN_RPI_RPI1_DMA_BASE;
clk_base = RCIN_RPI_RPI1_CLK_BASE; clk_base = RCIN_RPI_RPI1_CLK_BASE;
pcm_base = RCIN_RPI_RPI1_PCM_BASE; pcm_base = RCIN_RPI_RPI1_PCM_BASE;
} else if (_version == 1 || _version == 2) { } else if (_version == LINUX_BOARD_TYPE::RPI_2_3_ZERO2) {
// 2 & 3 are the same // 2 & 3 are the same
dma_base = RCIN_RPI_RPI2_DMA_BASE; dma_base = RCIN_RPI_RPI2_DMA_BASE;
clk_base = RCIN_RPI_RPI2_CLK_BASE; clk_base = RCIN_RPI_RPI2_CLK_BASE;
pcm_base = RCIN_RPI_RPI2_PCM_BASE; pcm_base = RCIN_RPI_RPI2_PCM_BASE;
} else if (_version == 3) { } else if (_version == LINUX_BOARD_TYPE::RPI_4) {
dma_base = RCIN_RPI_RPI4_DMA_BASE; dma_base = RCIN_RPI_RPI4_DMA_BASE;
clk_base = RCIN_RPI_RPI4_CLK_BASE; clk_base = RCIN_RPI_RPI4_CLK_BASE;
pcm_base = RCIN_RPI_RPI4_PCM_BASE; pcm_base = RCIN_RPI_RPI4_PCM_BASE;
} else { } else {
fprintf(stderr,"unknown RPI _version=%i\n", _version); fprintf(stderr,"Unknown Linux Board version!\n");
exit(-1); exit(-1);
} }
} }
@ -424,10 +424,11 @@ void RCInput_RPI::init_PCM()
hal.scheduler->delay_microseconds(100); hal.scheduler->delay_microseconds(100);
clk_reg[RCIN_RPI_PCMCLK_CNTL] = 0x5A000006; // Source=PLLD (500MHz) clk_reg[RCIN_RPI_PCMCLK_CNTL] = 0x5A000006; // Source=PLLD (500MHz)
hal.scheduler->delay_microseconds(100); hal.scheduler->delay_microseconds(100);
if (_version != 3) { if (_version != LINUX_BOARD_TYPE::RPI_4) {
clk_reg[RCIN_RPI_PCMCLK_DIV] = 0x5A000000 | ((RCIN_RPI_PLL_CLK/RCIN_RPI_SAMPLE_FREQ)<<12); // Set pcm div for BCM2835 500MHZ clock. If we need to configure DMA frequency. clk_reg[RCIN_RPI_PCMCLK_DIV] = 0x5A000000 | ((RCIN_RPI_PLL_CLK/RCIN_RPI_SAMPLE_FREQ)<<12); // Set pcm div for BCM2835 500MHZ clock. If we need to configure DMA frequency.
} }
else { else {
// RPI-4
clk_reg[RCIN_RPI_PCMCLK_DIV] = 0x5A000000 | ((RCIN_RPI4_PLL_CLK/RCIN_RPI_SAMPLE_FREQ)<< 12); // Set pcm div for BCM2711 700MHz clock. If we need to configure DMA frequency. clk_reg[RCIN_RPI_PCMCLK_DIV] = 0x5A000000 | ((RCIN_RPI4_PLL_CLK/RCIN_RPI_SAMPLE_FREQ)<< 12); // Set pcm div for BCM2711 700MHz clock. If we need to configure DMA frequency.
} }
hal.scheduler->delay_microseconds(100); hal.scheduler->delay_microseconds(100);
@ -524,9 +525,9 @@ void RCInput_RPI::init()
uint64_t signal_states(0); uint64_t signal_states(0);
#if CONFIG_HAL_BOARD_SUBTYPE == HAL_BOARD_SUBTYPE_LINUX_ERLEBRAIN2 #if CONFIG_HAL_BOARD_SUBTYPE == HAL_BOARD_SUBTYPE_LINUX_ERLEBRAIN2
_version = 1; _version = LINUX_BOARD_TYPE::RPI_2_3_ZERO2;
#else #else
_version = UtilRPI::from(hal.util)->get_rpi_version(); _version = UtilRPI::from(hal.util)->detect_linux_board_type();
#endif #endif
set_physical_addresses(); set_physical_addresses();

View File

@ -20,6 +20,8 @@
#include <assert.h> #include <assert.h>
#include <queue> #include <queue>
#include "Util_RPI.h"
#if CONFIG_HAL_BOARD_SUBTYPE == HAL_BOARD_SUBTYPE_LINUX_BH #if CONFIG_HAL_BOARD_SUBTYPE == HAL_BOARD_SUBTYPE_LINUX_BH
#define RCIN_RPI_CHN_NUM 8 #define RCIN_RPI_CHN_NUM 8
#elif CONFIG_HAL_BOARD_SUBTYPE == HAL_BOARD_SUBTYPE_LINUX_OBAL_V1 #elif CONFIG_HAL_BOARD_SUBTYPE == HAL_BOARD_SUBTYPE_LINUX_OBAL_V1
@ -61,7 +63,7 @@ private:
public: public:
Memory_table(); Memory_table();
Memory_table(uint32_t, int); Memory_table(const uint32_t, const LINUX_BOARD_TYPE);
~Memory_table(); ~Memory_table();
//Get virtual address from the corresponding physical address from memory_table. //Get virtual address from the corresponding physical address from memory_table.
@ -132,7 +134,7 @@ private:
} rc_channels[RCIN_RPI_CHN_NUM]; } rc_channels[RCIN_RPI_CHN_NUM];
bool _initialized = false; bool _initialized = false;
int _version =0; LINUX_BOARD_TYPE _version = LINUX_BOARD_TYPE::UNKNOWN_BOARD;
void init_dma_cb(dma_cb_t** cbp, uint32_t mode, uint32_t source, uint32_t dest, uint32_t length, uint32_t stride, uint32_t next_cb); void init_dma_cb(dma_cb_t** cbp, uint32_t mode, uint32_t source, uint32_t dest, uint32_t length, uint32_t stride, uint32_t next_cb);
void* map_peripheral(uint32_t base, uint32_t len); void* map_peripheral(uint32_t base, uint32_t len);

View File

@ -27,47 +27,67 @@ using namespace Linux;
UtilRPI::UtilRPI() UtilRPI::UtilRPI()
{ {
_check_rpi_version_by_rev(); _get_board_type_using_peripheral_base ();
}
int UtilRPI::_check_rpi_version_by_rev()
{
// assume 2 if unknown
_rpi_version = 2;
const char *SOC[]= {"Broadcom BCM2835","Broadcom BCM2836","Broadcom BCM2837","Broadcom BCM2711"};
const char *revision_file_ = "/proc/device-tree/system/linux,revision";
uint8_t revision[4] = { 0 };
uint32_t cpu = 0;
FILE *fd;
if ((fd = fopen(revision_file_, "rb")) == NULL) {
printf("Can't open '%s'\n", revision_file_);
}
else {
if (fread(revision, 1, sizeof(revision), fd) == 4) {
cpu = (revision[2] >> 4) & 0xf;
_rpi_version = cpu;
if (_rpi_version==0) {
_rpi_version=1; //RPI-Zero
}
printf("SOC: %s use intern: %d \r\n",SOC[cpu], _rpi_version);
}
else {
printf("Revision data too short\n");
}
fclose(fd);
}
return _rpi_version;
} }
int UtilRPI::get_rpi_version() const //
// previous appraoch was using /proc/device-tree/system/linux,revision
// now we use /proc/device-tree/soc/ranges see: https://forums.raspberrypi.com//viewtopic.php?t=244031
void UtilRPI::_get_board_type_using_peripheral_base()
{ {
return _rpi_version; FILE *fp;
uint32_t base=0x00;
unsigned char buf[32];
_linux_board_version = LINUX_BOARD_TYPE::UNKNOWN_BOARD;
fp = fopen("/proc/device-tree/soc/ranges" , "rb");
if (fp) {
const uint16_t len = fread(buf, 1, sizeof(buf), fp);
if (len >= 8) {
base = buf[4]<<24 | buf[5]<<16 | buf[6]<<8 | buf[7];
if (!base)
base = buf[8]<<24 | buf[9]<<16 | buf[10]<<8 | buf[11];
if (!base)
base = buf[8]<<24 | buf[9]<<16 | buf[10]<<8 | buf[11];
if (!base && (len>15))
base = buf[12]<<24 | buf[13]<<16 | buf[14]<<8 | buf[15];
}
fclose(fp);
}
switch (base) {
case 0x0:
_linux_board_version = LINUX_BOARD_TYPE::UNKNOWN_BOARD;
printf("Cannot detect board-type \r\n");
break;
case 0x20000000:
_linux_board_version = LINUX_BOARD_TYPE::RPI_ZERO_1;
printf("RPI Zero / 1 \r\n");
printf("Peripheral base address is %x\n", base);
break;
case 0x3f000000:
_linux_board_version = LINUX_BOARD_TYPE::RPI_2_3_ZERO2;
printf("RPI 2, 3 or Zero-2 \r\n");
printf("Peripheral base address is %x\n", base);
break;
case 0xfe000000:
_linux_board_version = LINUX_BOARD_TYPE::RPI_4;
printf("RPI 4 \r\n");
printf("Peripheral base address is %x\n", base);
break;
case 0x40000000:
_linux_board_version = LINUX_BOARD_TYPE::ALLWINNWER_H616;
printf("AllWinner-H616 \r\n");
break;
}
return ;
}
LINUX_BOARD_TYPE UtilRPI::detect_linux_board_type() const
{
return _linux_board_version;
} }
#endif #endif

View File

@ -4,6 +4,14 @@
namespace Linux { namespace Linux {
enum class LINUX_BOARD_TYPE: int {
RPI_ZERO_1=0,
RPI_2_3_ZERO2=1,
RPI_4=2,
ALLWINNWER_H616=100,
UNKNOWN_BOARD=999
};
class UtilRPI : public Util { class UtilRPI : public Util {
public: public:
UtilRPI(); UtilRPI();
@ -13,14 +21,14 @@ public:
} }
/* return the Raspberry Pi version */ /* return the Raspberry Pi version */
int get_rpi_version() const; LINUX_BOARD_TYPE detect_linux_board_type() const;
protected: protected:
// Called in the constructor once // Called in the constructor once
int _check_rpi_version_by_rev(); void _get_board_type_using_peripheral_base();
private: private:
int _rpi_version = 0; LINUX_BOARD_TYPE _linux_board_version = LINUX_BOARD_TYPE::UNKNOWN_BOARD;
}; };
} }