2015-08-11 03:28:43 -03:00
# include <AP_HAL/AP_HAL.h>
2014-10-26 16:18:04 -03:00
2015-11-28 05:38:56 -04:00
# if CONFIG_HAL_BOARD_SUBTYPE == HAL_BOARD_SUBTYPE_LINUX_NAVIO || \
CONFIG_HAL_BOARD_SUBTYPE = = HAL_BOARD_SUBTYPE_LINUX_ERLEBRAIN2 | | \
CONFIG_HAL_BOARD_SUBTYPE = = HAL_BOARD_SUBTYPE_LINUX_BH
2015-04-16 12:18:14 -03:00
# include "GPIO.h"
2015-11-02 11:21:10 -04:00
# include "RCInput_RPI.h"
2015-09-23 22:53:42 -03:00
# include "Util_RPI.h"
2015-09-23 12:52:07 -03:00
2014-10-26 16:18:04 -03:00
# include <stdio.h>
# include <stdlib.h>
# include <unistd.h>
# include <fcntl.h>
# include <stdint.h>
# include <sys/ioctl.h>
2015-04-16 12:18:14 -03:00
# include <pthread.h>
# include <string.h>
# include <errno.h>
# include <stdint.h>
# include <time.h>
# include <sys/time.h>
# include <sys/types.h>
# include <sys/stat.h>
# include <sys/mman.h>
# include <assert.h>
2015-09-23 12:52:07 -03:00
2014-10-26 16:18:04 -03:00
2015-04-16 12:18:14 -03:00
//Parametres
2015-11-02 11:21:10 -04:00
# define RCIN_RPI_BUFFER_LENGTH 8
# define RCIN_RPI_SAMPLE_FREQ 500
# define RCIN_RPI_DMA_CHANNEL 0
# define RCIN_RPI_MAX_SIZE_LINE 50
2015-04-16 12:18:14 -03:00
2015-11-28 05:38:56 -04:00
# if CONFIG_HAL_BOARD_SUBTYPE == HAL_BOARD_SUBTYPE_LINUX_BH
// Same as the circle_buffer size
// See comments in "init_ctrl_data()" to understand values "2"
# define RCIN_RPI_MAX_COUNTER (RCIN_RPI_BUFFER_LENGTH * PAGE_SIZE * 2)
# define RCIN_RPI_SIG_HIGH 0
# define RCIN_RPI_SIG_LOW 1
// Each gpio stands for a rcinput channel,
// the first one in RcChnGpioTbl is channel 1 in receiver
static uint16_t RcChnGpioTbl [ RCIN_RPI_CHN_NUM ] = {
RPI_GPIO_5 , RPI_GPIO_6 , RPI_GPIO_12 ,
RPI_GPIO_13 , RPI_GPIO_19 , RPI_GPIO_20 ,
RPI_GPIO_21 , RPI_GPIO_26
} ;
# else
# define RCIN_RPI_MAX_COUNTER 1304
# define RCIN_RPI_SIG_HIGH 1
# define RCIN_RPI_SIG_LOW 0
static uint16_t RcChnGpioTbl [ RCIN_RPI_CHN_NUM ] = {
RPI_GPIO_4
} ;
# endif // CONFIG_HAL_BOARD_SUBTYPE
2015-04-16 12:18:14 -03:00
//Memory Addresses
2015-11-02 11:21:10 -04:00
# define RCIN_RPI_RPI1_DMA_BASE 0x20007000
# define RCIN_RPI_RPI1_CLK_BASE 0x20101000
# define RCIN_RPI_RPI1_PCM_BASE 0x20203000
# define RCIN_RPI_RPI2_DMA_BASE 0x3F007000
# define RCIN_RPI_RPI2_CLK_BASE 0x3F101000
# define RCIN_RPI_RPI2_PCM_BASE 0x3F203000
# define RCIN_RPI_GPIO_LEV0_ADDR 0x7e200034
# define RCIN_RPI_DMA_LEN 0x1000
# define RCIN_RPI_CLK_LEN 0xA8
# define RCIN_RPI_PCM_LEN 0x24
# define RCIN_RPI_TIMER_BASE 0x7e003004
# define RCIN_RPI_DMA_SRC_INC (1<<8)
# define RCIN_RPI_DMA_DEST_INC (1<<4)
# define RCIN_RPI_DMA_NO_WIDE_BURSTS (1<<26)
# define RCIN_RPI_DMA_WAIT_RESP (1<<3)
# define RCIN_RPI_DMA_D_DREQ (1<<6)
# define RCIN_RPI_DMA_PER_MAP(x) ((x)<<16)
# define RCIN_RPI_DMA_END (1<<1)
# define RCIN_RPI_DMA_RESET (1<<31)
# define RCIN_RPI_DMA_INT (1<<2)
# define RCIN_RPI_DMA_CS (0x00 / 4)
# define RCIN_RPI_DMA_CONBLK_AD (0x04 / 4)
# define RCIN_RPI_DMA_DEBUG (0x20 / 4)
# define RCIN_RPI_PCM_CS_A (0x00 / 4)
# define RCIN_RPI_PCM_FIFO_A (0x04 / 4)
# define RCIN_RPI_PCM_MODE_A (0x08 / 4)
# define RCIN_RPI_PCM_RXC_A (0x0c / 4)
# define RCIN_RPI_PCM_TXC_A (0x10 / 4)
# define RCIN_RPI_PCM_DREQ_A (0x14 / 4)
# define RCIN_RPI_PCM_INTEN_A (0x18 / 4)
# define RCIN_RPI_PCM_INT_STC_A (0x1c / 4)
# define RCIN_RPI_PCM_GRAY (0x20 / 4)
# define RCIN_RPI_PCMCLK_CNTL 38
# define RCIN_RPI_PCMCLK_DIV 39
2015-04-16 12:18:14 -03:00
2014-10-26 16:18:04 -03:00
extern const AP_HAL : : HAL & hal ;
using namespace Linux ;
2015-11-02 11:21:10 -04:00
volatile uint32_t * RCInput_RPI : : pcm_reg ;
volatile uint32_t * RCInput_RPI : : clk_reg ;
volatile uint32_t * RCInput_RPI : : dma_reg ;
2015-04-16 12:18:14 -03:00
Memory_table : : Memory_table ( )
2014-10-26 16:18:04 -03:00
{
2015-04-16 12:18:14 -03:00
_page_count = 0 ;
}
//Init Memory table
Memory_table : : Memory_table ( uint32_t page_count , int version )
{
uint32_t i ;
int fdMem , file ;
//Cache coherent adresses depends on RPI's version
uint32_t bus = version = = 1 ? 0x40000000 : 0xC0000000 ;
uint64_t pageInfo ;
void * offset ;
_virt_pages = ( void * * ) malloc ( page_count * sizeof ( void * ) ) ;
_phys_pages = ( void * * ) malloc ( page_count * sizeof ( void * ) ) ;
_page_count = page_count ;
if ( ( fdMem = open ( " /dev/mem " , O_RDWR | O_SYNC ) ) < 0 ) {
fprintf ( stderr , " Failed to open /dev/mem \n " ) ;
exit ( - 1 ) ;
}
if ( ( file = open ( " /proc/self/pagemap " , O_RDWR | O_SYNC ) ) < 0 ) {
fprintf ( stderr , " Failed to open /proc/self/pagemap \n " ) ;
exit ( - 1 ) ;
}
//Magic to determine the physical address for this page:
offset = mmap ( 0 , _page_count * PAGE_SIZE , PROT_READ | PROT_WRITE , MAP_SHARED | MAP_ANONYMOUS | MAP_NORESERVE | MAP_LOCKED , - 1 , 0 ) ;
2015-08-15 07:29:14 -03:00
lseek ( file , ( ( uintptr_t ) offset ) / PAGE_SIZE * 8 , SEEK_SET ) ;
2015-04-16 12:18:14 -03:00
//Get list of available cache coherent physical addresses
for ( i = 0 ; i < _page_count ; i + + ) {
_virt_pages [ i ] = mmap ( 0 , PAGE_SIZE , PROT_READ | PROT_WRITE , MAP_SHARED | MAP_ANONYMOUS | MAP_NORESERVE | MAP_LOCKED , - 1 , 0 ) ;
: : read ( file , & pageInfo , 8 ) ;
2015-08-15 07:29:14 -03:00
_phys_pages [ i ] = ( void * ) ( ( uintptr_t ) ( pageInfo * PAGE_SIZE ) | bus ) ;
2015-04-16 12:18:14 -03:00
}
//Map physical addresses to virtual memory
for ( i = 0 ; i < _page_count ; i + + ) {
munmap ( _virt_pages [ i ] , PAGE_SIZE ) ;
2015-08-15 07:29:14 -03:00
_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 = = 1 ? 0xFFFFFFFF : ~ bus ) ) ) ;
2015-04-16 12:18:14 -03:00
memset ( _virt_pages [ i ] , 0xee , PAGE_SIZE ) ;
}
close ( file ) ;
close ( fdMem ) ;
}
Memory_table : : ~ Memory_table ( )
{
free ( _virt_pages ) ;
free ( _phys_pages ) ;
}
2015-07-01 09:04:57 -03:00
// This function returns physical address with help of pointer, which is offset from the beginning of the buffer.
void * Memory_table : : get_page ( void * * const pages , uint32_t addr ) const
2015-04-16 12:18:14 -03:00
{
2015-07-01 09:04:57 -03:00
if ( addr > = PAGE_SIZE * _page_count ) {
return NULL ;
2015-04-16 12:18:14 -03:00
}
2015-07-01 09:04:57 -03:00
return ( uint8_t * ) pages [ ( uint32_t ) addr / 4096 ] + addr % 4096 ;
2015-04-16 12:18:14 -03:00
}
2015-07-01 09:04:57 -03:00
//Get virtual address from the corresponding physical address from memory_table.
void * Memory_table : : get_virt_addr ( const uint32_t phys_addr ) const
2015-04-16 12:18:14 -03:00
{
2015-07-01 09:04:57 -03:00
// FIXME: Can't the address be calculated directly?
// FIXME: if the address room in _phys_pages is not fragmented one may avoid a complete loop ..
uint32_t i = 0 ;
for ( ; i < _page_count ; i + + ) {
2015-08-15 07:29:14 -03:00
if ( ( uintptr_t ) _phys_pages [ i ] = = ( ( ( uintptr_t ) phys_addr ) & 0xFFFFF000 ) ) {
return ( void * ) ( ( uintptr_t ) _virt_pages [ i ] + ( phys_addr & 0xFFF ) ) ;
2015-04-16 12:18:14 -03:00
}
}
return NULL ;
}
2015-07-01 09:04:57 -03:00
// FIXME: in-congruent function style see above
2015-04-16 12:18:14 -03:00
// This function returns offset from the beginning of the buffer using virtual address and memory_table.
2015-07-01 09:04:57 -03:00
uint32_t Memory_table : : get_offset ( void * * const pages , const uint32_t addr ) const
2015-04-16 12:18:14 -03:00
{
2015-07-01 09:04:57 -03:00
uint32_t i = 0 ;
for ( ; i < _page_count ; i + + ) {
2015-08-15 07:29:14 -03:00
if ( ( uintptr_t ) pages [ i ] = = ( addr & 0xFFFFF000 ) ) {
2015-07-01 09:04:57 -03:00
return ( i * PAGE_SIZE + ( addr & 0xFFF ) ) ;
2015-04-16 12:18:14 -03:00
}
}
return - 1 ;
}
//How many bytes are available for reading in circle buffer?
2015-07-01 09:04:57 -03:00
uint32_t Memory_table : : bytes_available ( const uint32_t read_addr , const uint32_t write_addr ) const
2015-04-16 12:18:14 -03:00
{
if ( write_addr > read_addr ) {
2015-07-01 09:04:57 -03:00
return ( write_addr - read_addr ) ;
2015-04-16 12:18:14 -03:00
}
else {
2015-07-01 09:04:57 -03:00
return _page_count * PAGE_SIZE - ( read_addr - write_addr ) ;
2015-04-16 12:18:14 -03:00
}
}
2015-07-01 09:04:57 -03:00
uint32_t Memory_table : : get_page_count ( ) const
2015-04-16 12:18:14 -03:00
{
return _page_count ;
}
//Physical addresses of peripheral depends on Raspberry Pi's version
2015-11-02 11:21:10 -04:00
void RCInput_RPI : : set_physical_addresses ( int version )
2015-04-16 12:18:14 -03:00
{
if ( version = = 1 ) {
2015-11-02 11:21:10 -04:00
dma_base = RCIN_RPI_RPI1_DMA_BASE ;
clk_base = RCIN_RPI_RPI1_CLK_BASE ;
pcm_base = RCIN_RPI_RPI1_PCM_BASE ;
2015-04-16 12:18:14 -03:00
}
else if ( version = = 2 ) {
2015-11-02 11:21:10 -04:00
dma_base = RCIN_RPI_RPI2_DMA_BASE ;
clk_base = RCIN_RPI_RPI2_CLK_BASE ;
pcm_base = RCIN_RPI_RPI2_PCM_BASE ;
2015-04-16 12:18:14 -03:00
}
}
//Map peripheral to virtual memory
2015-11-02 11:21:10 -04:00
void * RCInput_RPI : : map_peripheral ( uint32_t base , uint32_t len )
2015-04-16 12:18:14 -03:00
{
int fd = open ( " /dev/mem " , O_RDWR ) ;
void * vaddr ;
if ( fd < 0 ) {
printf ( " Failed to open /dev/mem: %m \n " ) ;
return NULL ;
}
vaddr = mmap ( NULL , len , PROT_READ | PROT_WRITE , MAP_SHARED , fd , base ) ;
if ( vaddr = = MAP_FAILED ) {
printf ( " rpio-pwm: Failed to map peripheral at 0x%08x: %m \n " , base ) ;
}
close ( fd ) ;
return vaddr ;
}
//Method to init DMA control block
2015-11-02 11:21:10 -04:00
void RCInput_RPI : : 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 )
2015-04-16 12:18:14 -03:00
{
( * cbp ) - > info = mode ;
( * cbp ) - > src = source ;
( * cbp ) - > dst = dest ;
( * cbp ) - > length = length ;
( * cbp ) - > next = next_cb ;
( * cbp ) - > stride = stride ;
}
2015-11-02 11:21:10 -04:00
void RCInput_RPI : : stop_dma ( )
2015-04-16 12:18:14 -03:00
{
2015-11-02 11:21:10 -04:00
dma_reg [ RCIN_RPI_DMA_CS | RCIN_RPI_DMA_CHANNEL < < 8 ] = 0 ;
2015-07-09 11:07:53 -03:00
}
/* We need to be sure that the DMA is stopped upon termination */
2015-11-02 11:21:10 -04:00
void RCInput_RPI : : termination_handler ( int signum )
2015-07-09 11:07:53 -03:00
{
stop_dma ( ) ;
2015-11-19 23:10:58 -04:00
AP_HAL : : panic ( " Interrupted " ) ;
2015-04-16 12:18:14 -03:00
}
//This function is used to init DMA control blocks (setting sampling GPIO register, destination adresses, synchronization)
2015-11-02 11:21:10 -04:00
void RCInput_RPI : : init_ctrl_data ( )
2015-04-16 12:18:14 -03:00
{
uint32_t phys_fifo_addr ;
uint32_t dest = 0 ;
uint32_t cbp = 0 ;
dma_cb_t * cbp_curr ;
//Set fifo addr (for delay)
phys_fifo_addr = ( ( pcm_base + 0x04 ) & 0x00FFFFFF ) | 0x7e000000 ;
2014-10-28 11:37:25 -03:00
2015-04-16 12:18:14 -03:00
//Init dma control blocks.
2015-11-28 05:38:56 -04:00
/*We are transferring 8 bytes of GPIO register. Every 7th iteration we are
sampling TIMER register , which length is 8 bytes . So , for every 7 samples of GPIO we need
7 * 8 + 8 = 64 bytes of buffer . Value 7 was selected specially to have a 64 - byte " block "
TIMER - GPIO . So , we have integer count of such " blocks " at one virtual page . ( 4096 / 64 = 64
" blocks " per page . As minimum , we must have 2 virtual pages of buffer ( to have integer count of
vitual pages for control blocks ) : for every 7 iterations ( 64 bytes of buffer ) we need 7 control blocks for GPIO
sampling , 7 control blocks for setting frequency and 1 control block for sampling timer , so ,
we need 7 + 7 + 1 = 15 control blocks . For integer value , we need 15 pages of control blocks .
Each control block length is 32 bytes . In 15 pages we will have ( 15 * 4096 / 32 ) = 15 * 128 control
blocks . 15 * 128 control blocks = 64 * 128 bytes of buffer = 2 pages of buffer .
So , for 7 * 64 * 2 iteration we init DMA for sampling GPIO
and timer to ( ( 7 * 8 + 8 ) * 64 * 2 ) = 8192 bytes = 2 pages of buffer .
2015-04-16 12:18:14 -03:00
*/
// fprintf(stderr, "ERROR SEARCH1\n");
2015-07-01 09:04:57 -03:00
uint32_t i = 0 ;
2015-11-28 05:38:56 -04:00
for ( i = 0 ; i < 7 * 128 * RCIN_RPI_BUFFER_LENGTH ; i + + ) // 7 * 128 * 8 == 7168
2015-04-16 12:18:14 -03:00
{
2015-11-28 05:38:56 -04:00
//Transfer timer every 7th sample
if ( i % 7 = = 0 ) {
2015-07-01 09:04:57 -03:00
cbp_curr = ( dma_cb_t * ) con_blocks - > get_page ( con_blocks - > _virt_pages , cbp ) ;
2015-04-16 12:18:14 -03:00
2015-11-02 11:21:10 -04:00
init_dma_cb ( & cbp_curr , RCIN_RPI_DMA_NO_WIDE_BURSTS | RCIN_RPI_DMA_WAIT_RESP | RCIN_RPI_DMA_DEST_INC | RCIN_RPI_DMA_SRC_INC , RCIN_RPI_TIMER_BASE ,
2015-08-15 07:29:14 -03:00
( uintptr_t ) circle_buffer - > get_page ( circle_buffer - > _phys_pages , dest ) ,
8 ,
0 ,
( uintptr_t ) con_blocks - > get_page ( con_blocks - > _phys_pages ,
cbp + sizeof ( dma_cb_t ) ) ) ;
2015-07-01 09:04:57 -03:00
dest + = 8 ;
cbp + = sizeof ( dma_cb_t ) ;
}
2015-04-16 12:18:14 -03:00
2015-11-28 05:38:56 -04:00
// Transfer GPIO (8 byte)
2015-07-01 09:04:57 -03:00
cbp_curr = ( dma_cb_t * ) con_blocks - > get_page ( con_blocks - > _virt_pages , cbp ) ;
2015-11-28 05:38:56 -04:00
init_dma_cb ( & cbp_curr , RCIN_RPI_DMA_NO_WIDE_BURSTS | RCIN_RPI_DMA_WAIT_RESP , RCIN_RPI_GPIO_LEV0_ADDR ,
( uintptr_t ) circle_buffer - > get_page ( circle_buffer - > _phys_pages , dest ) ,
8 ,
0 ,
( uintptr_t ) con_blocks - > get_page ( con_blocks - > _phys_pages ,
2015-08-15 07:29:14 -03:00
cbp + sizeof ( dma_cb_t ) ) ) ;
2015-11-28 05:38:56 -04:00
dest + = 8 ;
2015-04-16 12:18:14 -03:00
cbp + = sizeof ( dma_cb_t ) ;
// Delay (for setting sampling frequency)
/* DMA is waiting data request signal (DREQ) from PCM. PCM is set for 1 MhZ freqency, so,
each sample of GPIO is limited by writing to PCA queue .
*/
2015-07-01 09:04:57 -03:00
cbp_curr = ( dma_cb_t * ) con_blocks - > get_page ( con_blocks - > _virt_pages , cbp ) ;
2015-11-02 11:21:10 -04:00
init_dma_cb ( & cbp_curr , RCIN_RPI_DMA_NO_WIDE_BURSTS | RCIN_RPI_DMA_WAIT_RESP | RCIN_RPI_DMA_D_DREQ | RCIN_RPI_DMA_PER_MAP ( 2 ) ,
RCIN_RPI_TIMER_BASE , phys_fifo_addr ,
2015-08-15 07:29:14 -03:00
4 ,
0 ,
( uintptr_t ) con_blocks - > get_page ( con_blocks - > _phys_pages ,
cbp + sizeof ( dma_cb_t ) ) ) ;
2015-04-16 12:18:14 -03:00
cbp + = sizeof ( dma_cb_t ) ;
}
//Make last control block point to the first (to make circle)
cbp - = sizeof ( dma_cb_t ) ;
2015-08-15 07:29:14 -03:00
( ( dma_cb_t * ) con_blocks - > get_page ( con_blocks - > _virt_pages , cbp ) ) - > next = ( uintptr_t ) con_blocks - > get_page ( con_blocks - > _phys_pages , 0 ) ;
2015-04-16 12:18:14 -03:00
}
/*Initialise PCM
See BCM2835 documentation :
http : //www.raspberrypi.org/wp-content/uploads/2012/02/BCM2835-ARM-Peripherals.pdf
*/
2015-11-02 11:21:10 -04:00
void RCInput_RPI : : init_PCM ( )
2015-04-16 12:18:14 -03:00
{
2015-11-02 11:21:10 -04:00
pcm_reg [ RCIN_RPI_PCM_CS_A ] = 1 ; // Disable Rx+Tx, Enable PCM block
2015-04-16 12:18:14 -03:00
hal . scheduler - > delay_microseconds ( 100 ) ;
2015-11-02 11:21:10 -04:00
clk_reg [ RCIN_RPI_PCMCLK_CNTL ] = 0x5A000006 ; // Source=PLLD (500MHz)
2015-04-16 12:18:14 -03:00
hal . scheduler - > delay_microseconds ( 100 ) ;
2015-11-02 11:21:10 -04:00
clk_reg [ RCIN_RPI_PCMCLK_DIV ] = 0x5A000000 | ( ( 50000 / RCIN_RPI_SAMPLE_FREQ ) < < 12 ) ; // Set pcm div. If we need to configure DMA frequency.
2015-04-16 12:18:14 -03:00
hal . scheduler - > delay_microseconds ( 100 ) ;
2015-11-02 11:21:10 -04:00
clk_reg [ RCIN_RPI_PCMCLK_CNTL ] = 0x5A000016 ; // Source=PLLD and enable
2015-04-16 12:18:14 -03:00
hal . scheduler - > delay_microseconds ( 100 ) ;
2015-11-02 11:21:10 -04:00
pcm_reg [ RCIN_RPI_PCM_TXC_A ] = 0 < < 31 | 1 < < 30 | 0 < < 20 | 0 < < 16 ; // 1 channel, 8 bits
2015-04-16 12:18:14 -03:00
hal . scheduler - > delay_microseconds ( 100 ) ;
2015-11-02 11:21:10 -04:00
pcm_reg [ RCIN_RPI_PCM_MODE_A ] = ( 10 - 1 ) < < 10 ; //PCM mode
2015-04-16 12:18:14 -03:00
hal . scheduler - > delay_microseconds ( 100 ) ;
2015-11-02 11:21:10 -04:00
pcm_reg [ RCIN_RPI_PCM_CS_A ] | = 1 < < 4 | 1 < < 3 ; // Clear FIFOs
2015-04-16 12:18:14 -03:00
hal . scheduler - > delay_microseconds ( 100 ) ;
2015-11-02 11:21:10 -04:00
pcm_reg [ RCIN_RPI_PCM_DREQ_A ] = 64 < < 24 | 64 < < 8 ; // DMA Req when one slot is free?
2015-04-16 12:18:14 -03:00
hal . scheduler - > delay_microseconds ( 100 ) ;
2015-11-02 11:21:10 -04:00
pcm_reg [ RCIN_RPI_PCM_CS_A ] | = 1 < < 9 ; // Enable DMA
2015-04-16 12:18:14 -03:00
hal . scheduler - > delay_microseconds ( 100 ) ;
2015-11-02 11:21:10 -04:00
pcm_reg [ RCIN_RPI_PCM_CS_A ] | = 1 < < 2 ; // Enable Tx
2015-04-16 12:18:14 -03:00
hal . scheduler - > delay_microseconds ( 100 ) ;
}
/*Initialise DMA
See BCM2835 documentation :
http : //www.raspberrypi.org/wp-content/uploads/2012/02/BCM2835-ARM-Peripherals.pdf
*/
2015-11-02 11:21:10 -04:00
void RCInput_RPI : : init_DMA ( )
2015-04-16 12:18:14 -03:00
{
2015-11-02 11:21:10 -04:00
dma_reg [ RCIN_RPI_DMA_CS | RCIN_RPI_DMA_CHANNEL < < 8 ] = RCIN_RPI_DMA_RESET ; //Reset DMA
2015-04-16 12:18:14 -03:00
hal . scheduler - > delay_microseconds ( 100 ) ;
2015-11-02 11:21:10 -04:00
dma_reg [ RCIN_RPI_DMA_CS | RCIN_RPI_DMA_CHANNEL < < 8 ] = RCIN_RPI_DMA_INT | RCIN_RPI_DMA_END ;
dma_reg [ RCIN_RPI_DMA_CONBLK_AD | RCIN_RPI_DMA_CHANNEL < < 8 ] = reinterpret_cast < uintptr_t > ( con_blocks - > get_page ( con_blocks - > _phys_pages , 0 ) ) ; //Set first control block address
dma_reg [ RCIN_RPI_DMA_DEBUG | RCIN_RPI_DMA_CHANNEL < < 8 ] = 7 ; // clear debug error flags
dma_reg [ RCIN_RPI_DMA_CS | RCIN_RPI_DMA_CHANNEL < < 8 ] = 0x10880001 ; // go, mid priority, wait for outstanding writes
2015-04-16 12:18:14 -03:00
}
//We must stop DMA when the process is killed
2015-11-02 11:21:10 -04:00
void RCInput_RPI : : set_sigaction ( )
2015-04-16 12:18:14 -03:00
{
for ( int i = 0 ; i < 64 ; i + + ) {
//catch all signals (like ctrl+c, ctrl+z, ...) to ensure DMA is disabled
struct sigaction sa ;
memset ( & sa , 0 , sizeof ( sa ) ) ;
2015-11-02 11:21:10 -04:00
sa . sa_handler = RCInput_RPI : : termination_handler ;
2015-04-16 12:18:14 -03:00
sigaction ( i , & sa , NULL ) ;
}
}
//Initial setup of variables
2015-11-02 11:21:10 -04:00
RCInput_RPI : : RCInput_RPI ( ) :
curr_tick_inc ( 1000 / RCIN_RPI_SAMPLE_FREQ ) ,
2015-04-16 12:18:14 -03:00
curr_pointer ( 0 ) ,
2015-11-28 05:38:56 -04:00
curr_channel ( 0 )
2015-11-02 11:21:10 -04:00
{
# if CONFIG_HAL_BOARD_SUBTYPE == HAL_BOARD_SUBTYPE_LINUX_ERLEBRAIN2
int version = 2 ;
# else
2015-10-20 18:13:25 -03:00
int version = UtilRPI : : from ( hal . util ) - > get_rpi_version ( ) ;
2015-11-02 11:21:10 -04:00
# endif
2015-04-16 12:18:14 -03:00
set_physical_addresses ( version ) ;
2015-11-28 05:38:56 -04:00
//Init memory for buffer and for DMA control blocks. See comments in "init_ctrl_data()" to understand values "2" and "15"
2015-11-02 11:21:10 -04:00
circle_buffer = new Memory_table ( RCIN_RPI_BUFFER_LENGTH * 2 , version ) ;
2015-11-28 05:38:56 -04:00
con_blocks = new Memory_table ( RCIN_RPI_BUFFER_LENGTH * 15 , version ) ;
2015-04-16 12:18:14 -03:00
}
2015-11-02 11:21:10 -04:00
RCInput_RPI : : ~ RCInput_RPI ( )
2015-04-16 12:18:14 -03:00
{
delete circle_buffer ;
delete con_blocks ;
}
2015-11-02 11:21:10 -04:00
void RCInput_RPI : : deinit ( )
2015-07-06 19:49:04 -03:00
{
2015-07-09 11:07:53 -03:00
stop_dma ( ) ;
2015-07-06 19:49:04 -03:00
}
2015-04-16 12:18:14 -03:00
//Initializing necessary registers
2015-11-02 11:21:10 -04:00
void RCInput_RPI : : init_registers ( )
2015-04-16 12:18:14 -03:00
{
2015-11-02 11:21:10 -04:00
dma_reg = ( uint32_t * ) map_peripheral ( dma_base , RCIN_RPI_DMA_LEN ) ;
pcm_reg = ( uint32_t * ) map_peripheral ( pcm_base , RCIN_RPI_PCM_LEN ) ;
clk_reg = ( uint32_t * ) map_peripheral ( clk_base , RCIN_RPI_CLK_LEN ) ;
2015-04-16 12:18:14 -03:00
}
2015-12-02 11:14:20 -04:00
void RCInput_RPI : : init ( )
2015-04-16 12:18:14 -03:00
{
2015-11-28 05:38:56 -04:00
uint64_t signal_states ( 0 ) ;
2015-04-16 12:18:14 -03:00
init_registers ( ) ;
2015-11-28 05:38:56 -04:00
//Enable PPM or PWM input
for ( uint32_t i = 0 ; i < RCIN_RPI_CHN_NUM ; + + i ) {
rc_channels [ i ] . enable_pin = hal . gpio - > channel ( RcChnGpioTbl [ i ] ) ;
rc_channels [ i ] . enable_pin - > mode ( HAL_GPIO_INPUT ) ;
}
2015-04-16 12:18:14 -03:00
//Configuration
set_sigaction ( ) ;
init_ctrl_data ( ) ;
init_PCM ( ) ;
init_DMA ( ) ;
//wait a bit to let DMA fill queues and come to stable sampling
hal . scheduler - > delay ( 300 ) ;
//Reading first sample
2015-07-01 09:04:57 -03:00
curr_tick = * ( ( uint64_t * ) circle_buffer - > get_page ( circle_buffer - > _virt_pages , curr_pointer ) ) ;
2015-04-16 12:18:14 -03:00
curr_pointer + = 8 ;
2015-11-28 05:38:56 -04:00
signal_states = * ( ( uint64_t * ) circle_buffer - > get_page ( circle_buffer - > _virt_pages , curr_pointer ) ) ;
for ( uint32_t i = 0 ; i < RCIN_RPI_CHN_NUM ; + + i ) {
rc_channels [ i ] . prev_tick = curr_tick ;
rc_channels [ i ] . curr_signal = ( signal_states & ( 1 < < RcChnGpioTbl [ i ] ) ) ? RCIN_RPI_SIG_HIGH
: RCIN_RPI_SIG_LOW ;
rc_channels [ i ] . last_signal = rc_channels [ i ] . curr_signal ;
}
curr_pointer + = 8 ;
set_num_channels ( RCIN_RPI_CHN_NUM ) ;
2014-10-26 16:18:04 -03:00
}
2015-04-16 12:18:14 -03:00
//Processing signal
2015-11-02 11:21:10 -04:00
void RCInput_RPI : : _timer_tick ( )
2014-10-26 16:18:04 -03:00
{
2015-04-16 12:18:14 -03:00
int j ;
void * x ;
2015-11-28 05:38:56 -04:00
uint64_t signal_states ( 0 ) ;
2015-04-16 12:18:14 -03:00
//Now we are getting address in which DMAC is writing at current moment
2015-11-02 11:21:10 -04:00
dma_cb_t * ad = ( dma_cb_t * ) con_blocks - > get_virt_addr ( dma_reg [ RCIN_RPI_DMA_CONBLK_AD | RCIN_RPI_DMA_CHANNEL < < 8 ] ) ;
2015-04-16 12:18:14 -03:00
for ( j = 1 ; j > = - 1 ; j - - ) {
2015-07-01 09:04:57 -03:00
x = circle_buffer - > get_virt_addr ( ( ad + j ) - > dst ) ;
if ( x ! = NULL ) {
break ; }
2015-04-16 12:18:14 -03:00
}
//How many bytes have DMA transfered (and we can process)?
2015-08-15 07:29:14 -03:00
counter = circle_buffer - > bytes_available ( curr_pointer , circle_buffer - > get_offset ( circle_buffer - > _virt_pages , ( uintptr_t ) x ) ) ;
2015-04-16 12:18:14 -03:00
//We can't stay in method for a long time, because it may lead to delays
2015-11-02 11:21:10 -04:00
if ( counter > RCIN_RPI_MAX_COUNTER ) {
counter = RCIN_RPI_MAX_COUNTER ;
2015-04-16 12:18:14 -03:00
}
//Processing ready bytes
2015-11-28 05:38:56 -04:00
for ( ; counter > 0x40 ; ) {
2015-04-16 12:18:14 -03:00
//Is it timer samle?
if ( curr_pointer % ( 64 ) = = 0 ) {
2015-07-01 09:04:57 -03:00
curr_tick = * ( ( uint64_t * ) circle_buffer - > get_page ( circle_buffer - > _virt_pages , curr_pointer ) ) ;
2015-04-16 12:18:14 -03:00
curr_pointer + = 8 ;
counter - = 8 ;
2014-10-28 11:37:25 -03:00
}
2015-04-16 12:18:14 -03:00
//Reading required bit
2015-11-28 05:38:56 -04:00
signal_states = * ( ( uint64_t * ) circle_buffer - > get_page ( circle_buffer - > _virt_pages , curr_pointer ) ) ;
for ( uint32_t i = 0 ; i < RCIN_RPI_CHN_NUM ; + + i ) {
rc_channels [ i ] . curr_signal = ( signal_states & ( 1 < < RcChnGpioTbl [ i ] ) ) ? RCIN_RPI_SIG_HIGH
: RCIN_RPI_SIG_LOW ;
//If the signal changed
if ( rc_channels [ i ] . curr_signal ! = rc_channels [ i ] . last_signal ) {
rc_channels [ i ] . delta_time = curr_tick - rc_channels [ i ] . prev_tick ;
rc_channels [ i ] . prev_tick = curr_tick ;
switch ( rc_channels [ i ] . state ) {
case RCIN_RPI_INITIAL_STATE :
rc_channels [ i ] . state = RCIN_RPI_ZERO_STATE ;
break ;
case RCIN_RPI_ZERO_STATE :
if ( rc_channels [ i ] . curr_signal = = 0 ) {
rc_channels [ i ] . width_s0 = ( uint16_t ) rc_channels [ i ] . delta_time ;
rc_channels [ i ] . state = RCIN_RPI_ONE_STATE ;
break ;
}
else {
break ;
}
case RCIN_RPI_ONE_STATE :
if ( rc_channels [ i ] . curr_signal = = 1 ) {
rc_channels [ i ] . width_s1 = ( uint16_t ) rc_channels [ i ] . delta_time ;
rc_channels [ i ] . state = RCIN_RPI_ZERO_STATE ;
# if CONFIG_HAL_BOARD_SUBTYPE == HAL_BOARD_SUBTYPE_LINUX_BH
_process_rc_pulse ( rc_channels [ i ] . width_s0 ,
rc_channels [ i ] . width_s1 , i ) ;
# else
_process_rc_pulse ( rc_channels [ i ] . width_s0 ,
rc_channels [ i ] . width_s1 ) ;
# endif // CONFIG_HAL_BOARD_SUBTYPE
break ;
}
else {
break ;
}
2015-04-16 12:18:14 -03:00
}
}
2015-11-28 05:38:56 -04:00
rc_channels [ i ] . last_signal = rc_channels [ i ] . curr_signal ;
2015-04-16 12:18:14 -03:00
}
2015-11-28 05:38:56 -04:00
curr_pointer + = 8 ;
counter - = 8 ;
2015-04-16 12:18:14 -03:00
if ( curr_pointer > = circle_buffer - > get_page_count ( ) * PAGE_SIZE ) {
curr_pointer = 0 ;
}
curr_tick + = curr_tick_inc ;
}
2014-10-26 16:18:04 -03:00
}
# endif // CONFIG_HAL_BOARD_SUBTYPE