mirror of https://github.com/joan2937/pigpio
14045 lines
333 KiB
C
14045 lines
333 KiB
C
/*
|
|
This is free and unencumbered software released into the public domain.
|
|
|
|
Anyone is free to copy, modify, publish, use, compile, sell, or
|
|
distribute this software, either in source code form or as a compiled
|
|
binary, for any purpose, commercial or non-commercial, and by any
|
|
means.
|
|
|
|
In jurisdictions that recognize copyright laws, the author or authors
|
|
of this software dedicate any and all copyright interest in the
|
|
software to the public domain. We make this dedication for the benefit
|
|
of the public at large and to the detriment of our heirs and
|
|
successors. We intend this dedication to be an overt act of
|
|
relinquishment in perpetuity of all present and future rights to this
|
|
software under copyright law.
|
|
|
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
|
EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
|
|
IN NO EVENT SHALL THE AUTHORS BE LIABLE FOR ANY CLAIM, DAMAGES OR
|
|
OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
|
|
ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
|
|
OTHER DEALINGS IN THE SOFTWARE.
|
|
|
|
For more information, please refer to <http://unlicense.org/>
|
|
*/
|
|
|
|
/* pigpio version 80 */
|
|
|
|
/* include ------------------------------------------------------- */
|
|
|
|
#define _GNU_SOURCE
|
|
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include <strings.h>
|
|
#include <stdlib.h>
|
|
#include <stdint.h>
|
|
#include <inttypes.h>
|
|
#include <stdarg.h>
|
|
#include <ctype.h>
|
|
#include <syslog.h>
|
|
#include <poll.h>
|
|
#include <unistd.h>
|
|
#include <fcntl.h>
|
|
#include <termios.h>
|
|
#include <signal.h>
|
|
#include <errno.h>
|
|
#include <time.h>
|
|
#include <sys/ioctl.h>
|
|
#include <limits.h>
|
|
#include <pthread.h>
|
|
#include <sys/time.h>
|
|
#include <sys/mman.h>
|
|
#include <sys/types.h>
|
|
#include <sys/stat.h>
|
|
#include <sys/file.h>
|
|
#include <sys/socket.h>
|
|
#include <sys/sysmacros.h>
|
|
#include <netinet/tcp.h>
|
|
#include <arpa/inet.h>
|
|
#include <sys/select.h>
|
|
#include <fnmatch.h>
|
|
#include <glob.h>
|
|
#include <arpa/inet.h>
|
|
|
|
#include "pigpio.h"
|
|
|
|
#include "command.h"
|
|
|
|
|
|
/* --------------------------------------------------------------- */
|
|
|
|
/*
|
|
0 GPFSEL0 GPIO Function Select 0
|
|
1 GPFSEL1 GPIO Function Select 1
|
|
2 GPFSEL2 GPIO Function Select 2
|
|
3 GPFSEL3 GPIO Function Select 3
|
|
4 GPFSEL4 GPIO Function Select 4
|
|
5 GPFSEL5 GPIO Function Select 5
|
|
6 - Reserved
|
|
7 GPSET0 GPIO Pin Output Set 0
|
|
8 GPSET1 GPIO Pin Output Set 1
|
|
9 - Reserved
|
|
10 GPCLR0 GPIO Pin Output Clear 0
|
|
11 GPCLR1 GPIO Pin Output Clear 1
|
|
12 - Reserved
|
|
13 GPLEV0 GPIO Pin Level 0
|
|
14 GPLEV1 GPIO Pin Level 1
|
|
15 - Reserved
|
|
16 GPEDS0 GPIO Pin Event Detect Status 0
|
|
17 GPEDS1 GPIO Pin Event Detect Status 1
|
|
18 - Reserved
|
|
19 GPREN0 GPIO Pin Rising Edge Detect Enable 0
|
|
20 GPREN1 GPIO Pin Rising Edge Detect Enable 1
|
|
21 - Reserved
|
|
22 GPFEN0 GPIO Pin Falling Edge Detect Enable 0
|
|
23 GPFEN1 GPIO Pin Falling Edge Detect Enable 1
|
|
24 - Reserved
|
|
25 GPHEN0 GPIO Pin High Detect Enable 0
|
|
26 GPHEN1 GPIO Pin High Detect Enable 1
|
|
27 - Reserved
|
|
28 GPLEN0 GPIO Pin Low Detect Enable 0
|
|
29 GPLEN1 GPIO Pin Low Detect Enable 1
|
|
30 - Reserved
|
|
31 GPAREN0 GPIO Pin Async. Rising Edge Detect 0
|
|
32 GPAREN1 GPIO Pin Async. Rising Edge Detect 1
|
|
33 - Reserved
|
|
34 GPAFEN0 GPIO Pin Async. Falling Edge Detect 0
|
|
35 GPAFEN1 GPIO Pin Async. Falling Edge Detect 1
|
|
36 - Reserved
|
|
37 GPPUD GPIO Pin Pull-up/down Enable
|
|
38 GPPUDCLK0 GPIO Pin Pull-up/down Enable Clock 0
|
|
39 GPPUDCLK1 GPIO Pin Pull-up/down Enable Clock 1
|
|
40 - Reserved
|
|
41 - Test
|
|
42-56 Reserved
|
|
57 GPPUPPDN1 Pin pull-up/down for pins 15:0
|
|
58 GPPUPPDN1 Pin pull-up/down for pins 31:16
|
|
59 GPPUPPDN2 Pin pull-up/down for pins 47:32
|
|
60 GPPUPPDN3 Pin pull-up/down for pins 57:48
|
|
*/
|
|
|
|
/*
|
|
0 CS DMA Channel 0 Control and Status
|
|
1 CPI_ONBLK_AD DMA Channel 0 Control Block Address
|
|
2 TI DMA Channel 0 CB Word 0 (Transfer Information)
|
|
3 SOURCE_AD DMA Channel 0 CB Word 1 (Source Address)
|
|
4 DEST_AD DMA Channel 0 CB Word 2 (Destination Address)
|
|
5 TXFR_LEN DMA Channel 0 CB Word 3 (Transfer Length)
|
|
6 STRIDE DMA Channel 0 CB Word 4 (2D Stride)
|
|
7 NEXTCPI_ONBK DMA Channel 0 CB Word 5 (Next CB Address)
|
|
8 DEBUG DMA Channel 0 Debug
|
|
*/
|
|
|
|
/*
|
|
DEBUG register bits
|
|
|
|
bit 2 READ_ERROR
|
|
|
|
Slave Read Response Error RW 0x0
|
|
|
|
Set if the read operation returned an error value on
|
|
the read response bus. It can be cleared by writing
|
|
a 1.
|
|
|
|
bit 1 FIFO_ERROR
|
|
|
|
Fifo Error RW 0x0
|
|
|
|
Set if the optional read Fifo records an error
|
|
condition. It can be cleared by writing a 1.
|
|
|
|
bit 0 READ_LAST_NOT_SET_ERROR
|
|
|
|
Read Last Not Set Error RW 0x0
|
|
|
|
If the AXI read last signal was not set when
|
|
expected, then this error bit will be set. It can be
|
|
cleared by writing a 1.
|
|
*/
|
|
|
|
/*
|
|
0 CTL PWM Control
|
|
1 STA PWM Status
|
|
2 DMAC PWM DMA Configuration
|
|
4 RNG1 PWM Channel 1 Range
|
|
5 DAT1 PWM Channel 1 Data
|
|
6 FIF1 PWM FIFO Input
|
|
8 RNG2 PWM Channel 2 Range
|
|
9 DAT2 PWM Channel 2 Data
|
|
*/
|
|
|
|
/*
|
|
0 PCM_CS PCM Control and Status
|
|
1 PCM_FIFO PCM FIFO Data
|
|
2 PCM_MODE PCM Mode
|
|
3 PCM_RXC PCM Receive Configuration
|
|
4 PCM_TXC PCM Transmit Configuration
|
|
5 PCM_DREQ PCM DMA Request Level
|
|
6 PCM_INTEN PCM Interrupt Enables
|
|
7 PCM_INTSTC PCM Interrupt Status & Clear
|
|
8 PCM_GRAY PCM Gray Mode Control
|
|
*/
|
|
|
|
/*
|
|
0 CS System Timer Control/Status
|
|
1 CLO System Timer Counter Lower 32 bits
|
|
2 CHI System Timer Counter Higher 32 bits
|
|
3 C0 System Timer Compare 0
|
|
4 C1 System Timer Compare 1
|
|
5 C2 System Timer Compare 2
|
|
6 C3 System Timer Compare 3
|
|
*/
|
|
|
|
/* define -------------------------------------------------------- */
|
|
|
|
#define THOUSAND 1000
|
|
#define MILLION 1000000
|
|
#define BILLION 1000000000
|
|
|
|
#define BANK (gpio>>5)
|
|
|
|
#define BIT (1<<(gpio&0x1F))
|
|
|
|
#ifndef EMBEDDED_IN_VM
|
|
#define DBG(level, format, arg...) DO_DBG(level, format, ## arg)
|
|
#else
|
|
#define DBG(level, format, arg...)
|
|
#endif
|
|
|
|
#define DO_DBG(level, format, arg...) \
|
|
{ \
|
|
if ((gpioCfg.dbgLevel >= level) && \
|
|
(!(gpioCfg.internals & PI_CFG_NOSIGHANDLER))) \
|
|
fprintf(stderr, "%s %s: " format "\n" , \
|
|
myTimeStamp(), __FUNCTION__ , ## arg); \
|
|
}
|
|
|
|
#ifndef DISABLE_SER_CHECK_INITED
|
|
#define SER_CHECK_INITED CHECK_INITED
|
|
#else
|
|
#define SER_CHECK_INITED
|
|
#endif
|
|
|
|
#define CHECK_INITED \
|
|
do \
|
|
{ \
|
|
if (!libInitialised) \
|
|
{ \
|
|
DBG(DBG_ALWAYS, \
|
|
"pigpio uninitialised, call gpioInitialise()"); \
|
|
return PI_NOT_INITIALISED; \
|
|
} \
|
|
} \
|
|
while (0)
|
|
|
|
#define CHECK_INITED_RET_NULL_PTR \
|
|
do \
|
|
{ \
|
|
if (!libInitialised) \
|
|
{ \
|
|
DBG(DBG_ALWAYS, \
|
|
"pigpio uninitialised, call gpioInitialise()"); \
|
|
return (NULL); \
|
|
} \
|
|
} \
|
|
while (0)
|
|
|
|
#define CHECK_INITED_RET_NIL \
|
|
do \
|
|
{ \
|
|
if (!libInitialised) \
|
|
{ \
|
|
DBG(DBG_ALWAYS, \
|
|
"pigpio uninitialised, call gpioInitialise()"); \
|
|
} \
|
|
} \
|
|
while (0)
|
|
|
|
#define CHECK_NOT_INITED \
|
|
do \
|
|
{ \
|
|
if (libInitialised) \
|
|
{ \
|
|
DBG(DBG_ALWAYS, \
|
|
"pigpio initialised, call gpioTerminate()"); \
|
|
return PI_INITIALISED; \
|
|
} \
|
|
} \
|
|
while (0)
|
|
|
|
#define SOFT_ERROR(x, format, arg...) \
|
|
do \
|
|
{ \
|
|
DBG(DBG_ALWAYS, format, ## arg); \
|
|
return x; \
|
|
} \
|
|
while (0)
|
|
|
|
#define TIMER_ADD(a, b, result) \
|
|
do \
|
|
{ \
|
|
(result)->tv_sec = (a)->tv_sec + (b)->tv_sec; \
|
|
(result)->tv_nsec = (a)->tv_nsec + (b)->tv_nsec; \
|
|
if ((result)->tv_nsec >= BILLION) \
|
|
{ \
|
|
++(result)->tv_sec; \
|
|
(result)->tv_nsec -= BILLION; \
|
|
} \
|
|
} \
|
|
while (0)
|
|
|
|
#define TIMER_SUB(a, b, result) \
|
|
do \
|
|
{ \
|
|
(result)->tv_sec = (a)->tv_sec - (b)->tv_sec; \
|
|
(result)->tv_nsec = (a)->tv_nsec - (b)->tv_nsec; \
|
|
if ((result)->tv_nsec < 0) \
|
|
{ \
|
|
--(result)->tv_sec; \
|
|
(result)->tv_nsec += BILLION; \
|
|
} \
|
|
} \
|
|
while (0)
|
|
|
|
#define PI_PERI_BUS 0x7E000000
|
|
|
|
#define AUX_BASE (pi_peri_phys + 0x00215000)
|
|
#define BSCS_BASE (pi_peri_phys + 0x00214000)
|
|
#define CLK_BASE (pi_peri_phys + 0x00101000)
|
|
#define DMA_BASE (pi_peri_phys + 0x00007000)
|
|
#define DMA15_BASE (pi_peri_phys + 0x00E05000)
|
|
#define GPIO_BASE (pi_peri_phys + 0x00200000)
|
|
#define PADS_BASE (pi_peri_phys + 0x00100000)
|
|
#define PCM_BASE (pi_peri_phys + 0x00203000)
|
|
#define PWM_BASE (pi_peri_phys + 0x0020C000)
|
|
#define SPI_BASE (pi_peri_phys + 0x00204000)
|
|
#define SYST_BASE (pi_peri_phys + 0x00003000)
|
|
|
|
#define AUX_LEN 0xD8
|
|
#define BSCS_LEN 0x40
|
|
#define CLK_LEN 0xA8
|
|
#define DMA_LEN 0x1000 /* allow access to all channels */
|
|
#define GPIO_LEN 0xF4 /* 2711 has more registers */
|
|
#define PADS_LEN 0x38
|
|
#define PCM_LEN 0x24
|
|
#define PWM_LEN 0x28
|
|
#define SPI_LEN 0x18
|
|
#define SYST_LEN 0x1C
|
|
|
|
#define DMA_ENABLE (0xFF0/4)
|
|
|
|
#define GPFSEL0 0
|
|
|
|
#define GPSET0 7
|
|
#define GPSET1 8
|
|
|
|
#define GPCLR0 10
|
|
#define GPCLR1 11
|
|
|
|
#define GPLEV0 13
|
|
#define GPLEV1 14
|
|
|
|
#define GPEDS0 16
|
|
#define GPEDS1 17
|
|
|
|
#define GPREN0 19
|
|
#define GPREN1 20
|
|
#define GPFEN0 22
|
|
#define GPFEN1 23
|
|
#define GPHEN0 25
|
|
#define GPHEN1 26
|
|
#define GPLEN0 28
|
|
#define GPLEN1 29
|
|
#define GPAREN0 31
|
|
#define GPAREN1 32
|
|
#define GPAFEN0 34
|
|
#define GPAFEN1 35
|
|
|
|
#define GPPUD 37
|
|
#define GPPUDCLK0 38
|
|
#define GPPUDCLK1 39
|
|
|
|
/* BCM2711 has different pulls */
|
|
|
|
#define GPPUPPDN0 57
|
|
#define GPPUPPDN1 58
|
|
#define GPPUPPDN2 59
|
|
#define GPPUPPDN3 60
|
|
|
|
#define DMA_CS 0
|
|
#define DMA_CONBLK_AD 1
|
|
#define DMA_DEBUG 8
|
|
|
|
/* DMA CS Control and Status bits */
|
|
#define DMA_CHANNEL_RESET (1<<31)
|
|
#define DMA_CHANNEL_ABORT (1<<30)
|
|
#define DMA_WAIT_ON_WRITES (1<<28)
|
|
#define DMA_PANIC_PRIORITY(x) ((x)<<20)
|
|
#define DMA_PRIORITY(x) ((x)<<16)
|
|
#define DMA_INTERRUPT_STATUS (1<< 2)
|
|
#define DMA_END_FLAG (1<< 1)
|
|
#define DMA_ACTIVE (1<< 0)
|
|
|
|
/* DMA control block "info" field bits */
|
|
#define DMA_NO_WIDE_BURSTS (1<<26)
|
|
#define DMA_PERIPHERAL_MAPPING(x) ((x)<<16)
|
|
#define DMA_BURST_LENGTH(x) ((x)<<12)
|
|
#define DMA_SRC_IGNORE (1<<11)
|
|
#define DMA_SRC_DREQ (1<<10)
|
|
#define DMA_SRC_WIDTH (1<< 9)
|
|
#define DMA_SRC_INC (1<< 8)
|
|
#define DMA_DEST_IGNORE (1<< 7)
|
|
#define DMA_DEST_DREQ (1<< 6)
|
|
#define DMA_DEST_WIDTH (1<< 5)
|
|
#define DMA_DEST_INC (1<< 4)
|
|
#define DMA_WAIT_RESP (1<< 3)
|
|
#define DMA_TDMODE (1<< 1)
|
|
|
|
#define DMA_DEBUG_READ_ERR (1<<2)
|
|
#define DMA_DEBUG_FIFO_ERR (1<<1)
|
|
#define DMA_DEBUG_RD_LST_NOT_SET_ERR (1<<0)
|
|
|
|
#define DMA_LITE_FIRST 7
|
|
#define DMA_LITE_MAX 0xfffc
|
|
|
|
#define PWM_CTL 0
|
|
#define PWM_STA 1
|
|
#define PWM_DMAC 2
|
|
#define PWM_RNG1 4
|
|
#define PWM_DAT1 5
|
|
#define PWM_FIFO 6
|
|
#define PWM_RNG2 8
|
|
#define PWM_DAT2 9
|
|
|
|
#define PWM_CTL_MSEN2 (1<<15)
|
|
#define PWM_CTL_PWEN2 (1<<8)
|
|
#define PWM_CTL_MSEN1 (1<<7)
|
|
#define PWM_CTL_CLRF1 (1<<6)
|
|
#define PWM_CTL_USEF1 (1<<5)
|
|
#define PWM_CTL_MODE1 (1<<1)
|
|
#define PWM_CTL_PWEN1 (1<<0)
|
|
|
|
#define PWM_DMAC_ENAB (1 <<31)
|
|
#define PWM_DMAC_PANIC(x) ((x)<< 8)
|
|
#define PWM_DMAC_DREQ(x) (x)
|
|
|
|
#define PCM_CS 0
|
|
#define PCM_FIFO 1
|
|
#define PCM_MODE 2
|
|
#define PCM_RXC 3
|
|
#define PCM_TXC 4
|
|
#define PCM_DREQ 5
|
|
#define PCM_INTEN 6
|
|
#define PCM_INTSTC 7
|
|
#define PCM_GRAY 8
|
|
|
|
#define PCM_CS_STBY (1 <<25)
|
|
#define PCM_CS_SYNC (1 <<24)
|
|
#define PCM_CS_RXSEX (1 <<23)
|
|
#define PCM_CS_RXERR (1 <<16)
|
|
#define PCM_CS_TXERR (1 <<15)
|
|
#define PCM_CS_DMAEN (1 <<9)
|
|
#define PCM_CS_RXTHR(x) ((x)<<7)
|
|
#define PCM_CS_TXTHR(x) ((x)<<5)
|
|
#define PCM_CS_RXCLR (1 <<4)
|
|
#define PCM_CS_TXCLR (1 <<3)
|
|
#define PCM_CS_TXON (1 <<2)
|
|
#define PCM_CS_RXON (1 <<1)
|
|
#define PCM_CS_EN (1 <<0)
|
|
|
|
#define PCM_MODE_CLK_DIS (1 <<28)
|
|
#define PCM_MODE_PDMN (1 <<27)
|
|
#define PCM_MODE_PDME (1 <<26)
|
|
#define PCM_MODE_FRXP (1 <<25)
|
|
#define PCM_MODE_FTXP (1 <<24)
|
|
#define PCM_MODE_CLKM (1 <<23)
|
|
#define PCM_MODE_CLKI (1 <<22)
|
|
#define PCM_MODE_FSM (1 <<21)
|
|
#define PCM_MODE_FSI (1 <<20)
|
|
#define PCM_MODE_FLEN(x) ((x)<<10)
|
|
#define PCM_MODE_FSLEN(x) ((x)<< 0)
|
|
|
|
#define PCM_RXC_CH1WEX (1 <<31)
|
|
#define PCM_RXC_CH1EN (1 <<30)
|
|
#define PCM_RXC_CH1POS(x) ((x)<<20)
|
|
#define PCM_RXC_CH1WID(x) ((x)<<16)
|
|
#define PCM_RXC_CH2WEX (1 <<15)
|
|
#define PCM_RXC_CH2EN (1 <<14)
|
|
#define PCM_RXC_CH2POS(x) ((x)<< 4)
|
|
#define PCM_RXC_CH2WID(x) ((x)<< 0)
|
|
|
|
#define PCM_TXC_CH1WEX (1 <<31)
|
|
#define PCM_TXC_CH1EN (1 <<30)
|
|
#define PCM_TXC_CH1POS(x) ((x)<<20)
|
|
#define PCM_TXC_CH1WID(x) ((x)<<16)
|
|
#define PCM_TXC_CH2WEX (1 <<15)
|
|
#define PCM_TXC_CH2EN (1 <<14)
|
|
#define PCM_TXC_CH2POS(x) ((x)<< 4)
|
|
#define PCM_TXC_CH2WID(x) ((x)<< 0)
|
|
|
|
#define PCM_DREQ_TX_PANIC(x) ((x)<<24)
|
|
#define PCM_DREQ_RX_PANIC(x) ((x)<<16)
|
|
#define PCM_DREQ_TX_REQ_L(x) ((x)<< 8)
|
|
#define PCM_DREQ_RX_REQ_L(x) ((x)<< 0)
|
|
|
|
#define PCM_INTEN_RXERR (1<<3)
|
|
#define PCM_INTEN_TXERR (1<<2)
|
|
#define PCM_INTEN_RXR (1<<1)
|
|
#define PCM_INTEN_TXW (1<<0)
|
|
|
|
#define PCM_INTSTC_RXERR (1<<3)
|
|
#define PCM_INTSTC_TXERR (1<<2)
|
|
#define PCM_INTSTC_RXR (1<<1)
|
|
#define PCM_INTSTC_TXW (1<<0)
|
|
|
|
#define PCM_GRAY_FLUSH (1<<2)
|
|
#define PCM_GRAY_CLR (1<<1)
|
|
#define PCM_GRAY_EN (1<<0)
|
|
|
|
#define BCM_PASSWD (0x5A<<24)
|
|
|
|
#define CLK_CTL_MASH(x)((x)<<9)
|
|
#define CLK_CTL_BUSY (1 <<7)
|
|
#define CLK_CTL_KILL (1 <<5)
|
|
#define CLK_CTL_ENAB (1 <<4)
|
|
#define CLK_CTL_SRC(x) ((x)<<0)
|
|
|
|
#define CLK_SRCS 2
|
|
|
|
#define CLK_CTL_SRC_OSC 1
|
|
#define CLK_CTL_SRC_PLLD 6
|
|
|
|
#define CLK_OSC_FREQ 19200000
|
|
#define CLK_OSC_FREQ_2711 54000000
|
|
#define CLK_PLLD_FREQ 500000000
|
|
#define CLK_PLLD_FREQ_2711 750000000
|
|
|
|
#define CLK_DIV_DIVI(x) ((x)<<12)
|
|
#define CLK_DIV_DIVF(x) ((x)<< 0)
|
|
|
|
#define CLK_GP0_CTL 28
|
|
#define CLK_GP0_DIV 29
|
|
#define CLK_GP1_CTL 30
|
|
#define CLK_GP1_DIV 31
|
|
#define CLK_GP2_CTL 32
|
|
#define CLK_GP2_DIV 33
|
|
|
|
#define CLK_PCMCTL 38
|
|
#define CLK_PCMDIV 39
|
|
|
|
#define CLK_PWMCTL 40
|
|
#define CLK_PWMDIV 41
|
|
|
|
#define SYST_CS 0
|
|
#define SYST_CLO 1
|
|
#define SYST_CHI 2
|
|
|
|
/* SPI */
|
|
|
|
#define SPI_CS 0
|
|
#define SPI_FIFO 1
|
|
#define SPI_CLK 2
|
|
#define SPI_DLEN 3
|
|
#define SPI_LTOH 4
|
|
#define SPI_DC 5
|
|
|
|
#define SPI_CS_LEN_LONG (1<<25)
|
|
#define SPI_CS_DMA_LEN (1<<24)
|
|
#define SPI_CS_CSPOLS(x) ((x)<<21)
|
|
#define SPI_CS_RXF (1<<20)
|
|
#define SPI_CS_RXR (1<<19)
|
|
#define SPI_CS_TXD (1<<18)
|
|
#define SPI_CS_RXD (1<<17)
|
|
#define SPI_CS_DONE (1<<16)
|
|
#define SPI_CS_LEN (1<<13)
|
|
#define SPI_CS_REN (1<<12)
|
|
#define SPI_CS_ADCS (1<<11)
|
|
#define SPI_CS_INTR (1<<10)
|
|
#define SPI_CS_INTD (1<<9)
|
|
#define SPI_CS_DMAEN (1<<8)
|
|
#define SPI_CS_TA (1<<7)
|
|
#define SPI_CS_CSPOL(x) ((x)<<6)
|
|
#define SPI_CS_CLEAR(x) ((x)<<4)
|
|
#define SPI_CS_MODE(x) ((x)<<2)
|
|
#define SPI_CS_CS(x) ((x)<<0)
|
|
|
|
#define SPI_DC_RPANIC(x) ((x)<<24)
|
|
#define SPI_DC_RDREQ(x) ((x)<<16)
|
|
#define SPI_DC_TPANIC(x) ((x)<<8)
|
|
#define SPI_DC_TDREQ(x) ((x)<<0)
|
|
|
|
#define SPI_MODE0 0
|
|
#define SPI_MODE1 1
|
|
#define SPI_MODE2 2
|
|
#define SPI_MODE3 3
|
|
|
|
#define SPI_CS0 0
|
|
#define SPI_CS1 1
|
|
#define SPI_CS2 2
|
|
|
|
/* standard SPI gpios (ALT0) */
|
|
|
|
#define PI_SPI_CE0 8
|
|
#define PI_SPI_CE1 7
|
|
#define PI_SPI_SCLK 11
|
|
#define PI_SPI_MISO 9
|
|
#define PI_SPI_MOSI 10
|
|
|
|
/* auxiliary SPI gpios (ALT4) */
|
|
|
|
#define PI_ASPI_CE0 18
|
|
#define PI_ASPI_CE1 17
|
|
#define PI_ASPI_CE2 16
|
|
#define PI_ASPI_MISO 19
|
|
#define PI_ASPI_MOSI 20
|
|
#define PI_ASPI_SCLK 21
|
|
|
|
/* AUX */
|
|
|
|
#define AUX_IRQ 0
|
|
#define AUX_ENABLES 1
|
|
|
|
#define AUX_MU_IO_REG 16
|
|
#define AUX_MU_IER_REG 17
|
|
#define AUX_MU_IIR_REG 18
|
|
#define AUX_MU_LCR_REG 19
|
|
#define AUX_MU_MCR_REG 20
|
|
#define AUX_MU_LSR_REG 21
|
|
#define AUX_MU_MSR_REG 22
|
|
#define AUX_MU_SCRATCH 23
|
|
#define AUX_MU_CNTL_REG 24
|
|
#define AUX_MU_STAT_REG 25
|
|
#define AUX_MU_BAUD_REG 26
|
|
|
|
#define AUX_SPI0_CNTL0_REG 32
|
|
#define AUX_SPI0_CNTL1_REG 33
|
|
#define AUX_SPI0_STAT_REG 34
|
|
#define AUX_SPI0_PEEK_REG 35
|
|
|
|
#define AUX_SPI0_IO_REG 40
|
|
#define AUX_SPI0_TX_HOLD 44
|
|
|
|
#define AUX_SPI1_CNTL0_REG 48
|
|
#define AUX_SPI1_CNTL1_REG 49
|
|
#define AUX_SPI1_STAT_REG 50
|
|
#define AUX_SPI1_PEEK_REG 51
|
|
|
|
#define AUX_SPI1_IO_REG 56
|
|
#define AUX_SPI1_TX_HOLD 60
|
|
|
|
#define AUXENB_SPI2 (1<<2)
|
|
#define AUXENB_SPI1 (1<<1)
|
|
#define AUXENB_UART (1<<0)
|
|
|
|
#define AUXSPI_CNTL0_SPEED(x) ((x)<<20)
|
|
#define AUXSPI_CNTL0_CS(x) ((x)<<17)
|
|
#define AUXSPI_CNTL0_POSTINP (1<<16)
|
|
#define AUXSPI_CNTL0_VAR_CS (1<<15)
|
|
#define AUXSPI_CNTL0_VAR_WIDTH (1<<14)
|
|
#define AUXSPI_CNTL0_DOUT_HOLD(x) ((x)<<12)
|
|
#define AUXSPI_CNTL0_ENABLE (1<<11)
|
|
#define AUXSPI_CNTL0_IN_RISING(x) ((x)<<10)
|
|
#define AUXSPI_CNTL0_CLR_FIFOS (1<<9)
|
|
#define AUXSPI_CNTL0_OUT_RISING(x) ((x)<<8)
|
|
#define AUXSPI_CNTL0_INVERT_CLK(x) ((x)<<7)
|
|
#define AUXSPI_CNTL0_MSB_FIRST(x) ((x)<<6)
|
|
#define AUXSPI_CNTL0_SHIFT_LEN(x) ((x)<<0)
|
|
|
|
#define AUXSPI_CNTL1_CS_HIGH(x) ((x)<<8)
|
|
#define AUXSPI_CNTL1_TX_IRQ (1<<7)
|
|
#define AUXSPI_CNTL1_DONE_IRQ (1<<6)
|
|
#define AUXSPI_CNTL1_MSB_FIRST(x)((x)<<1)
|
|
#define AUXSPI_CNTL1_KEEP_INPUT (1<<0)
|
|
|
|
#define AUXSPI_STAT_TX_FIFO(x) ((x)<<28)
|
|
#define AUXSPI_STAT_RX_FIFO(x) ((x)<<20)
|
|
#define AUXSPI_STAT_TX_FULL (1<<10)
|
|
#define AUXSPI_STAT_TX_EMPTY (1<<9)
|
|
#define AUXSPI_STAT_RX_EMPTY (1<<7)
|
|
#define AUXSPI_STAT_BUSY (1<<6)
|
|
#define AUXSPI_STAT_BITS(x) ((x)<<0)
|
|
|
|
/* --------------------------------------------------------------- */
|
|
|
|
#define NORMAL_DMA (DMA_NO_WIDE_BURSTS | DMA_WAIT_RESP)
|
|
#define TWO_BEAT_DMA (DMA_TDMODE | DMA_BURST_LENGTH(1))
|
|
|
|
#define TIMED_DMA(x) (DMA_DEST_DREQ | DMA_PERIPHERAL_MAPPING(x))
|
|
|
|
#define PCM_TIMER (((PCM_BASE + PCM_FIFO*4) & 0x00ffffff) | PI_PERI_BUS)
|
|
#define PWM_TIMER (((PWM_BASE + PWM_FIFO*4) & 0x00ffffff) | PI_PERI_BUS)
|
|
|
|
#define DBG_MIN_LEVEL 0
|
|
#define DBG_ALWAYS 0
|
|
#define DBG_STARTUP 1
|
|
#define DBG_DMACBS 2
|
|
#define DBG_SCRIPT 3
|
|
#define DBG_USER 4
|
|
#define DBG_INTERNAL 5
|
|
#define DBG_SLOW_TICK 6
|
|
#define DBG_FAST_TICK 7
|
|
#define DBG_MAX_LEVEL 8
|
|
|
|
#define GPIO_UNDEFINED 0
|
|
#define GPIO_WRITE 1
|
|
#define GPIO_PWM 2
|
|
#define GPIO_SERVO 3
|
|
#define GPIO_HW_CLK 4
|
|
#define GPIO_HW_PWM 5
|
|
#define GPIO_SPI 6
|
|
#define GPIO_I2C 7
|
|
|
|
#define STACK_SIZE (256*1024)
|
|
|
|
#define PAGE_SIZE 4096
|
|
|
|
#define PWM_FREQS 18
|
|
|
|
#define CYCLES_PER_BLOCK 80
|
|
#define PULSE_PER_CYCLE 25
|
|
|
|
#define PAGES_PER_BLOCK 53
|
|
|
|
#define CBS_PER_IPAGE 117
|
|
#define LVS_PER_IPAGE 38
|
|
#define OFF_PER_IPAGE 38
|
|
#define TCK_PER_IPAGE 2
|
|
#define ON_PER_IPAGE 2
|
|
#define PAD_PER_IPAGE 7
|
|
|
|
#define CBS_PER_OPAGE 118
|
|
#define OOL_PER_OPAGE 79
|
|
|
|
/*
|
|
Wave Count Block
|
|
|
|
Assumes two counters per block. Each counter 4 * 16 (16^4=65536)
|
|
0 CB [13] 13*8 104 CBs for counter 0
|
|
104 CB [13] 13*8 104 CBs for counter 1
|
|
208 CB [60] 60*8 480 CBs reserved to construct wave
|
|
688 OOL [60] 60*1 60 OOL reserved to construct wave
|
|
748 OOL[136] 136*1 136 OOL for counter 0
|
|
884 OOL[136] 136*1 136 OOL for counter 1
|
|
1020 pad [4] 4*1 4 spare
|
|
*/
|
|
|
|
#define WCB_CNT_PER_PAGE 2
|
|
#define WCB_COUNTERS (WCB_CNT_PER_PAGE * PI_WAVE_COUNT_PAGES)
|
|
#define WCB_CNT_CBS 13
|
|
#define WCB_CNT_OOL 68
|
|
#define WCB_COUNTER_OOL (WCB_CNT_PER_PAGE * WCB_CNT_OOL)
|
|
#define WCB_COUNTER_CBS (WCB_CNT_PER_PAGE * WCB_CNT_CBS)
|
|
#define WCB_CHAIN_CBS 60
|
|
#define WCB_CHAIN_OOL 60
|
|
|
|
#define CBS_PER_CYCLE ((PULSE_PER_CYCLE*3)+2)
|
|
|
|
#define NUM_CBS (CBS_PER_CYCLE * bufferCycles)
|
|
|
|
#define SUPERCYCLE 800
|
|
#define SUPERLEVEL 20000
|
|
|
|
#define BLOCK_SIZE (PAGES_PER_BLOCK*PAGE_SIZE)
|
|
|
|
#define DMAI_PAGES (PAGES_PER_BLOCK * bufferBlocks)
|
|
|
|
#define DMAO_PAGES (PAGES_PER_BLOCK * PI_WAVE_BLOCKS)
|
|
|
|
#define NUM_WAVE_OOL (DMAO_PAGES * OOL_PER_OPAGE)
|
|
#define NUM_WAVE_CBS (DMAO_PAGES * CBS_PER_OPAGE)
|
|
|
|
#define TICKSLOTS 50
|
|
|
|
#define PI_I2C_CLOSED 0
|
|
#define PI_I2C_RESERVED 1
|
|
#define PI_I2C_OPENED 2
|
|
|
|
#define PI_SPI_CLOSED 0
|
|
#define PI_SPI_RESERVED 1
|
|
#define PI_SPI_OPENED 2
|
|
|
|
#define PI_SER_CLOSED 0
|
|
#define PI_SER_RESERVED 1
|
|
#define PI_SER_OPENED 2
|
|
|
|
#define PI_FILE_CLOSED 0
|
|
#define PI_FILE_RESERVED 1
|
|
#define PI_FILE_OPENED 2
|
|
|
|
#define PI_NOTIFY_CLOSED 0
|
|
#define PI_NOTIFY_RESERVED 1
|
|
#define PI_NOTIFY_CLOSING 2
|
|
#define PI_NOTIFY_OPENED 3
|
|
#define PI_NOTIFY_RUNNING 4
|
|
#define PI_NOTIFY_PAUSED 5
|
|
|
|
#define PI_WFRX_NONE 0
|
|
#define PI_WFRX_SERIAL 1
|
|
#define PI_WFRX_I2C_SDA 2
|
|
#define PI_WFRX_I2C_SCL 3
|
|
#define PI_WFRX_SPI_SCLK 4
|
|
#define PI_WFRX_SPI_MISO 5
|
|
#define PI_WFRX_SPI_MOSI 6
|
|
#define PI_WFRX_SPI_CS 7
|
|
|
|
#define PI_WF_MICROS 1
|
|
|
|
#define BPD 4
|
|
|
|
#define MAX_REPORT 250
|
|
#define MAX_SAMPLE 4000
|
|
|
|
#define DEFAULT_PWM_IDX 5
|
|
|
|
#define MAX_EMITS (PIPE_BUF / sizeof(gpioReport_t))
|
|
|
|
#define SRX_BUF_SIZE 8192
|
|
|
|
#define PI_I2C_RETRIES 0x0701
|
|
#define PI_I2C_TIMEOUT 0x0702
|
|
#define PI_I2C_SLAVE 0x0703
|
|
#define PI_I2C_FUNCS 0x0705
|
|
#define PI_I2C_RDWR 0x0707
|
|
#define PI_I2C_SMBUS 0x0720
|
|
|
|
#define PI_I2C_SMBUS_READ 1
|
|
#define PI_I2C_SMBUS_WRITE 0
|
|
|
|
#define PI_I2C_SMBUS_QUICK 0
|
|
#define PI_I2C_SMBUS_BYTE 1
|
|
#define PI_I2C_SMBUS_BYTE_DATA 2
|
|
#define PI_I2C_SMBUS_WORD_DATA 3
|
|
#define PI_I2C_SMBUS_PROC_CALL 4
|
|
#define PI_I2C_SMBUS_BLOCK_DATA 5
|
|
#define PI_I2C_SMBUS_I2C_BLOCK_BROKEN 6
|
|
#define PI_I2C_SMBUS_BLOCK_PROC_CALL 7
|
|
#define PI_I2C_SMBUS_I2C_BLOCK_DATA 8
|
|
|
|
#define PI_I2C_SMBUS_BLOCK_MAX 32
|
|
#define PI_I2C_SMBUS_I2C_BLOCK_MAX 32
|
|
|
|
#define PI_I2C_FUNC_SMBUS_QUICK 0x00010000
|
|
#define PI_I2C_FUNC_SMBUS_READ_BYTE 0x00020000
|
|
#define PI_I2C_FUNC_SMBUS_WRITE_BYTE 0x00040000
|
|
#define PI_I2C_FUNC_SMBUS_READ_BYTE_DATA 0x00080000
|
|
#define PI_I2C_FUNC_SMBUS_WRITE_BYTE_DATA 0x00100000
|
|
#define PI_I2C_FUNC_SMBUS_READ_WORD_DATA 0x00200000
|
|
#define PI_I2C_FUNC_SMBUS_WRITE_WORD_DATA 0x00400000
|
|
#define PI_I2C_FUNC_SMBUS_PROC_CALL 0x00800000
|
|
#define PI_I2C_FUNC_SMBUS_READ_BLOCK_DATA 0x01000000
|
|
#define PI_I2C_FUNC_SMBUS_WRITE_BLOCK_DATA 0x02000000
|
|
#define PI_I2C_FUNC_SMBUS_READ_I2C_BLOCK 0x04000000
|
|
#define PI_I2C_FUNC_SMBUS_WRITE_I2C_BLOCK 0x08000000
|
|
|
|
#define PI_MASH_MAX_FREQ 23800000
|
|
|
|
#define FLUSH_PAGES 1024
|
|
|
|
#define MB_DEV_MAJOR 100
|
|
|
|
#define MB_IOCTL _IOWR(MB_DEV_MAJOR, 0, char *)
|
|
|
|
#define MB_DEV1 "/dev/vcio"
|
|
#define MB_DEV2 "/dev/pigpio-mb"
|
|
|
|
#define BUS_TO_PHYS(x) ((x)&~0xC0000000)
|
|
|
|
#define MB_END_TAG 0
|
|
#define MB_PROCESS_REQUEST 0
|
|
|
|
#define MB_ALLOCATE_MEMORY_TAG 0x3000C
|
|
#define MB_LOCK_MEMORY_TAG 0x3000D
|
|
#define MB_UNLOCK_MEMORY_TAG 0x3000E
|
|
#define MB_RELEASE_MEMORY_TAG 0x3000F
|
|
|
|
#define PI_SCRIPT_FREE 0
|
|
#define PI_SCRIPT_RESERVED 1
|
|
#define PI_SCRIPT_IN_USE 2
|
|
#define PI_SCRIPT_DYING 3
|
|
|
|
#define PI_SCRIPT_HALT 0
|
|
#define PI_SCRIPT_RUN 1
|
|
#define PI_SCRIPT_DELETE 2
|
|
|
|
#define PI_SCRIPT_STACK_SIZE 256
|
|
|
|
#define PI_SPI_FLAGS_CHANNEL(x) ((x&7)<<29)
|
|
|
|
#define PI_SPI_FLAGS_GET_CHANNEL(x) (((x)>>29)&7)
|
|
#define PI_SPI_FLAGS_GET_BITLEN(x) (((x)>>16)&63)
|
|
#define PI_SPI_FLAGS_GET_RX_LSB(x) (((x)>>15)&1)
|
|
#define PI_SPI_FLAGS_GET_TX_LSB(x) (((x)>>14)&1)
|
|
#define PI_SPI_FLAGS_GET_3WREN(x) (((x)>>10)&15)
|
|
#define PI_SPI_FLAGS_GET_3WIRE(x) (((x)>>9)&1)
|
|
#define PI_SPI_FLAGS_GET_AUX_SPI(x) (((x)>>8)&1)
|
|
#define PI_SPI_FLAGS_GET_RESVD(x) (((x)>>5)&7)
|
|
#define PI_SPI_FLAGS_GET_CSPOLS(x) (((x)>>2)&7)
|
|
#define PI_SPI_FLAGS_GET_MODE(x) ((x)&3)
|
|
|
|
#define PI_SPI_FLAGS_GET_CPHA(x) ((x)&1)
|
|
#define PI_SPI_FLAGS_GET_CPOL(x) ((x)&2)
|
|
#define PI_SPI_FLAGS_GET_CSPOL(x) ((x)&4)
|
|
|
|
#define PI_STARTING 0
|
|
#define PI_RUNNING 1
|
|
#define PI_ENDING 2
|
|
|
|
#define PI_THREAD_NONE 0
|
|
#define PI_THREAD_STARTED 1
|
|
#define PI_THREAD_RUNNING 2
|
|
|
|
#define PI_MAX_PATH 512
|
|
|
|
/* typedef ------------------------------------------------------- */
|
|
|
|
typedef void (*callbk_t) ();
|
|
|
|
typedef struct
|
|
{
|
|
rawCbs_t cb [128];
|
|
} dmaPage_t;
|
|
|
|
typedef struct
|
|
{
|
|
rawCbs_t cb [CBS_PER_IPAGE];
|
|
uint32_t level [LVS_PER_IPAGE];
|
|
uint32_t gpioOff [OFF_PER_IPAGE];
|
|
uint32_t tick [TCK_PER_IPAGE];
|
|
uint32_t gpioOn [ON_PER_IPAGE];
|
|
uint32_t periphData;
|
|
uint32_t pad [PAD_PER_IPAGE];
|
|
} dmaIPage_t;
|
|
|
|
typedef struct
|
|
{
|
|
rawCbs_t cb [CBS_PER_OPAGE];
|
|
uint32_t OOL [OOL_PER_OPAGE];
|
|
uint32_t periphData;
|
|
} dmaOPage_t;
|
|
|
|
typedef struct
|
|
{
|
|
uint8_t is;
|
|
uint8_t pad;
|
|
uint16_t width;
|
|
uint16_t range; /* dutycycles specified by 0 .. range */
|
|
uint16_t freqIdx;
|
|
uint16_t deferOff;
|
|
uint16_t deferRng;
|
|
} gpioInfo_t;
|
|
|
|
typedef struct
|
|
{
|
|
callbk_t func;
|
|
unsigned ex;
|
|
void *userdata;
|
|
|
|
int wdSteadyUs;
|
|
uint32_t wdTick;
|
|
uint32_t wdLBitV;
|
|
|
|
int nfSteadyUs;
|
|
int nfActiveUs;
|
|
int nfActive;
|
|
uint32_t nfTick1;
|
|
uint32_t nfTick2;
|
|
uint32_t nfLBitV;
|
|
uint32_t nfRBitV;
|
|
|
|
uint32_t gfSteadyUs;
|
|
uint8_t gfInitialised;
|
|
uint32_t gfTick;
|
|
uint32_t gfLBitV;
|
|
uint32_t gfRBitV;
|
|
|
|
} gpioAlert_t;
|
|
|
|
typedef struct
|
|
{
|
|
callbk_t func;
|
|
unsigned ex;
|
|
void *userdata;
|
|
int ignore;
|
|
int fired;
|
|
} eventAlert_t;
|
|
|
|
typedef struct
|
|
{
|
|
unsigned gpio;
|
|
pthread_t *pth;
|
|
callbk_t func;
|
|
unsigned edge;
|
|
int timeout;
|
|
unsigned ex;
|
|
void *userdata;
|
|
int fd;
|
|
int inited;
|
|
} gpioISR_t;
|
|
|
|
typedef struct
|
|
{
|
|
callbk_t func;
|
|
unsigned ex;
|
|
void *userdata;
|
|
} gpioSignal_t;
|
|
|
|
typedef struct
|
|
{
|
|
callbk_t func;
|
|
unsigned ex;
|
|
void *userdata;
|
|
uint32_t bits;
|
|
} gpioGetSamples_t;
|
|
|
|
typedef struct
|
|
{
|
|
callbk_t func;
|
|
unsigned ex;
|
|
void *userdata;
|
|
unsigned id;
|
|
unsigned running;
|
|
unsigned millis;
|
|
pthread_t pthId;
|
|
} gpioTimer_t;
|
|
|
|
typedef struct
|
|
{
|
|
unsigned id;
|
|
unsigned state;
|
|
unsigned request;
|
|
unsigned run_state;
|
|
uint32_t waitBits;
|
|
uint32_t eventBits;
|
|
uint32_t changedBits;
|
|
pthread_t *pthIdp;
|
|
pthread_mutex_t pthMutex;
|
|
pthread_cond_t pthCond;
|
|
cmdScript_t script;
|
|
} gpioScript_t;
|
|
|
|
|
|
typedef struct
|
|
{
|
|
uint16_t valid;
|
|
uint16_t servoIdx;
|
|
} clkCfg_t;
|
|
|
|
typedef struct
|
|
{
|
|
uint16_t seqno;
|
|
uint16_t state;
|
|
uint32_t bits;
|
|
uint32_t eventBits;
|
|
uint32_t lastReportTick;
|
|
int fd;
|
|
int pipe;
|
|
int max_emits;
|
|
} gpioNotify_t;
|
|
|
|
typedef struct
|
|
{
|
|
uint16_t state;
|
|
int16_t fd;
|
|
uint32_t mode;
|
|
} fileInfo_t;
|
|
|
|
typedef struct
|
|
{
|
|
uint16_t state;
|
|
int16_t fd;
|
|
uint32_t addr;
|
|
uint32_t flags;
|
|
uint32_t funcs;
|
|
} i2cInfo_t;
|
|
|
|
typedef struct
|
|
{
|
|
uint16_t state;
|
|
int16_t fd;
|
|
uint32_t flags;
|
|
} serInfo_t;
|
|
|
|
typedef struct
|
|
{
|
|
uint16_t state;
|
|
unsigned speed;
|
|
uint32_t flags;
|
|
} spiInfo_t;
|
|
|
|
typedef struct
|
|
{
|
|
uint32_t alertTicks;
|
|
uint32_t lateTicks;
|
|
uint32_t moreToDo;
|
|
uint32_t diffTick[TICKSLOTS];
|
|
uint32_t cbTicks;
|
|
uint32_t cbCalls;
|
|
uint32_t maxEmit;
|
|
uint32_t emitFrags;
|
|
uint32_t maxSamples;
|
|
uint32_t numSamples;
|
|
uint32_t DMARestarts;
|
|
uint32_t dmaInitCbsCount;
|
|
uint32_t goodPipeWrite;
|
|
uint32_t shortPipeWrite;
|
|
uint32_t wouldBlockPipeWrite;
|
|
} gpioStats_t;
|
|
|
|
typedef struct
|
|
{
|
|
unsigned bufferMilliseconds;
|
|
unsigned clockMicros;
|
|
unsigned clockPeriph;
|
|
unsigned DMAprimaryChannel;
|
|
unsigned DMAsecondaryChannel;
|
|
unsigned socketPort;
|
|
unsigned ifFlags;
|
|
unsigned memAllocMode;
|
|
unsigned dbgLevel;
|
|
unsigned alertFreq;
|
|
uint32_t internals;
|
|
/*
|
|
0-3: dbgLevel
|
|
4-7: alertFreq
|
|
*/
|
|
} gpioCfg_t;
|
|
|
|
typedef struct
|
|
{
|
|
uint32_t micros;
|
|
uint32_t highMicros;
|
|
uint32_t maxMicros;
|
|
uint32_t pulses;
|
|
uint32_t highPulses;
|
|
uint32_t maxPulses;
|
|
uint32_t cbs;
|
|
uint32_t highCbs;
|
|
uint32_t maxCbs;
|
|
} wfStats_t;
|
|
|
|
typedef struct
|
|
{
|
|
char *buf;
|
|
uint32_t bufSize;
|
|
int readPos;
|
|
int writePos;
|
|
uint32_t fullBit; /* nanoseconds */
|
|
uint32_t halfBit; /* nanoseconds */
|
|
int timeout; /* millisconds */
|
|
uint32_t startBitTick; /* microseconds */
|
|
uint32_t nextBitDiff; /* nanoseconds */
|
|
int bit;
|
|
uint32_t data;
|
|
int bytes; /* 1, 2, 4 */
|
|
int level;
|
|
int dataBits; /* 1-32 */
|
|
int invert; /* 0, 1 */
|
|
} wfRxSerial_t;
|
|
|
|
typedef struct
|
|
{
|
|
int SDA;
|
|
int SCL;
|
|
int delay;
|
|
int SDAMode;
|
|
int SCLMode;
|
|
int started;
|
|
} wfRxI2C_t;
|
|
|
|
typedef struct
|
|
{
|
|
int CS;
|
|
int MISO;
|
|
int MOSI;
|
|
int SCLK;
|
|
int usage;
|
|
int delay;
|
|
int spiFlags;
|
|
int MISOMode;
|
|
int MOSIMode;
|
|
int CSMode;
|
|
int SCLKMode;
|
|
} wfRxSPI_t;
|
|
|
|
typedef struct
|
|
{
|
|
int mode;
|
|
int gpio;
|
|
uint32_t baud;
|
|
pthread_mutex_t mutex;
|
|
union
|
|
{
|
|
wfRxSerial_t s;
|
|
wfRxI2C_t I;
|
|
wfRxSPI_t S;
|
|
};
|
|
} wfRx_t;
|
|
|
|
union my_smbus_data
|
|
{
|
|
uint8_t byte;
|
|
uint16_t word;
|
|
uint8_t block[PI_I2C_SMBUS_BLOCK_MAX + 2];
|
|
};
|
|
|
|
struct my_smbus_ioctl_data
|
|
{
|
|
uint8_t read_write;
|
|
uint8_t command;
|
|
uint32_t size;
|
|
union my_smbus_data *data;
|
|
};
|
|
|
|
typedef struct
|
|
{
|
|
pi_i2c_msg_t *msgs; /* pointers to pi_i2c_msgs */
|
|
uint32_t nmsgs; /* number of pi_i2c_msgs */
|
|
} my_i2c_rdwr_ioctl_data_t;
|
|
|
|
typedef struct
|
|
{
|
|
unsigned div;
|
|
unsigned frac;
|
|
unsigned clock;
|
|
} clkInf_t;
|
|
|
|
typedef struct
|
|
{
|
|
unsigned handle; /* mbAllocateMemory() */
|
|
uintptr_t bus_addr; /* mbLockMemory() */
|
|
uintptr_t *virtual_addr; /* mbMapMem() */
|
|
unsigned size; /* in bytes */
|
|
} DMAMem_t;
|
|
|
|
/* global -------------------------------------------------------- */
|
|
|
|
/* initialise once then preserve */
|
|
|
|
static volatile uint32_t piCores = 0;
|
|
static volatile uint32_t pi_peri_phys = 0x20000000;
|
|
static volatile uint32_t pi_dram_bus = 0x40000000;
|
|
static volatile uint32_t pi_mem_flag = 0x0C;
|
|
static volatile uint32_t pi_ispi = 0;
|
|
static volatile uint32_t pi_is_2711 = 0;
|
|
static volatile uint32_t clk_osc_freq = CLK_OSC_FREQ;
|
|
static volatile uint32_t clk_plld_freq = CLK_PLLD_FREQ;
|
|
static volatile uint32_t hw_pwm_max_freq = PI_HW_PWM_MAX_FREQ;
|
|
static volatile uint32_t hw_clk_min_freq = PI_HW_CLK_MIN_FREQ;
|
|
static volatile uint32_t hw_clk_max_freq = PI_HW_CLK_MAX_FREQ;
|
|
|
|
static int libInitialised = 0;
|
|
|
|
/* initialise every gpioInitialise */
|
|
|
|
static struct timespec libStarted;
|
|
|
|
static uint32_t sockNetAddr[MAX_CONNECT_ADDRESSES];
|
|
|
|
static int numSockNetAddr = 0;
|
|
|
|
static uint32_t reportedLevel = 0;
|
|
|
|
static int waveClockInited = 0;
|
|
static int PWMClockInited = 0;
|
|
|
|
static volatile gpioStats_t gpioStats;
|
|
|
|
static int gpioMaskSet = 0;
|
|
|
|
/* initialise if not libInitialised */
|
|
|
|
static uint64_t gpioMask;
|
|
|
|
static rawWave_t wf[3][PI_WAVE_MAX_PULSES];
|
|
|
|
static int wfc[3]={0, 0, 0};
|
|
|
|
static int wfcur=0;
|
|
|
|
static wfStats_t wfStats=
|
|
{
|
|
0, 0, PI_WAVE_MAX_MICROS,
|
|
0, 0, PI_WAVE_MAX_PULSES,
|
|
0, 0, (DMAO_PAGES * CBS_PER_OPAGE)
|
|
};
|
|
|
|
static rawWaveInfo_t waveInfo[PI_MAX_WAVES];
|
|
|
|
static wfRx_t wfRx[PI_MAX_USER_GPIO+1];
|
|
|
|
static int waveOutBotCB = PI_WAVE_COUNT_PAGES*CBS_PER_OPAGE;
|
|
static int waveOutBotOOL = PI_WAVE_COUNT_PAGES*OOL_PER_OPAGE;
|
|
static int waveOutTopOOL = NUM_WAVE_OOL;
|
|
static int waveOutCount = 0;
|
|
|
|
static uint32_t *waveEndPtr = NULL;
|
|
|
|
static volatile uint32_t alertBits = 0;
|
|
static volatile uint32_t monitorBits = 0;
|
|
static volatile uint32_t notifyBits = 0;
|
|
static volatile uint32_t scriptBits = 0;
|
|
static volatile uint32_t gFilterBits = 0;
|
|
static volatile uint32_t nFilterBits = 0;
|
|
static volatile uint32_t wdogBits = 0;
|
|
|
|
static volatile uint32_t scriptEventBits = 0;
|
|
|
|
static volatile int runState = PI_STARTING;
|
|
|
|
static int pthAlertRunning = PI_THREAD_NONE;
|
|
static int pthFifoRunning = PI_THREAD_NONE;
|
|
static int pthSocketRunning = PI_THREAD_NONE;
|
|
|
|
static gpioAlert_t gpioAlert [PI_MAX_USER_GPIO+1];
|
|
|
|
static eventAlert_t eventAlert [PI_MAX_EVENT+1];
|
|
|
|
static gpioISR_t gpioISR [PI_MAX_GPIO+1];
|
|
|
|
static gpioGetSamples_t gpioGetSamples;
|
|
|
|
static gpioInfo_t gpioInfo [PI_MAX_GPIO+1];
|
|
|
|
static gpioNotify_t gpioNotify [PI_NOTIFY_SLOTS];
|
|
|
|
static fileInfo_t fileInfo [PI_FILE_SLOTS];
|
|
static i2cInfo_t i2cInfo [PI_I2C_SLOTS];
|
|
static serInfo_t serInfo [PI_SER_SLOTS];
|
|
static spiInfo_t spiInfo [PI_SPI_SLOTS];
|
|
|
|
static gpioScript_t gpioScript [PI_MAX_SCRIPTS];
|
|
|
|
static gpioSignal_t gpioSignal [PI_MAX_SIGNUM+1];
|
|
|
|
static gpioTimer_t gpioTimer [PI_MAX_TIMER+1];
|
|
|
|
static int pwmFreq[PWM_FREQS];
|
|
|
|
/* reset after gpioTerminated */
|
|
|
|
/* resources which must be released on gpioTerminate */
|
|
|
|
static FILE * inpFifo = NULL;
|
|
static FILE * outFifo = NULL;
|
|
|
|
static int fdLock = -1;
|
|
static int fdMem = -1;
|
|
static int fdSock = -1;
|
|
static int fdPmap = -1;
|
|
static int fdMbox = -1;
|
|
|
|
static DMAMem_t *dmaMboxBlk = MAP_FAILED;
|
|
static uintptr_t * * dmaPMapBlk = MAP_FAILED;
|
|
static dmaPage_t * * dmaVirt = MAP_FAILED;
|
|
static dmaPage_t * * dmaBus = MAP_FAILED;
|
|
|
|
static dmaIPage_t * * dmaIVirt = MAP_FAILED;
|
|
static dmaIPage_t * * dmaIBus = MAP_FAILED;
|
|
|
|
static dmaOPage_t * * dmaOVirt = MAP_FAILED;
|
|
static dmaOPage_t * * dmaOBus = MAP_FAILED;
|
|
|
|
static volatile uint32_t * auxReg = MAP_FAILED;
|
|
static volatile uint32_t * bscsReg = MAP_FAILED;
|
|
static volatile uint32_t * clkReg = MAP_FAILED;
|
|
static volatile uint32_t * dmaReg = MAP_FAILED;
|
|
static volatile uint32_t * gpioReg = MAP_FAILED;
|
|
static volatile uint32_t * padsReg = MAP_FAILED;
|
|
static volatile uint32_t * pcmReg = MAP_FAILED;
|
|
static volatile uint32_t * pwmReg = MAP_FAILED;
|
|
static volatile uint32_t * spiReg = MAP_FAILED;
|
|
static volatile uint32_t * systReg = MAP_FAILED;
|
|
|
|
static volatile uint32_t * dmaIn = MAP_FAILED;
|
|
static volatile uint32_t * dmaOut = MAP_FAILED;
|
|
|
|
static uint32_t hw_clk_freq[3];
|
|
static uint32_t hw_pwm_freq[2];
|
|
static uint32_t hw_pwm_duty[2];
|
|
static uint32_t hw_pwm_real_range[2];
|
|
|
|
static volatile gpioCfg_t gpioCfg =
|
|
{
|
|
PI_DEFAULT_BUFFER_MILLIS,
|
|
PI_DEFAULT_CLK_MICROS,
|
|
PI_DEFAULT_CLK_PERIPHERAL,
|
|
PI_DEFAULT_DMA_NOT_SET, /* primary DMA */
|
|
PI_DEFAULT_DMA_NOT_SET, /* secondary DMA */
|
|
PI_DEFAULT_SOCKET_PORT,
|
|
PI_DEFAULT_IF_FLAGS,
|
|
PI_DEFAULT_MEM_ALLOC_MODE,
|
|
0, /* dbgLevel */
|
|
0, /* alertFreq */
|
|
0, /* internals */
|
|
};
|
|
|
|
/* no initialisation required */
|
|
|
|
static unsigned bufferBlocks; /* number of blocks in buffer */
|
|
static unsigned bufferCycles; /* number of cycles */
|
|
|
|
static pthread_t pthAlert;
|
|
static pthread_t pthFifo;
|
|
static pthread_t pthSocket;
|
|
|
|
static uint32_t spi_dummy;
|
|
|
|
static unsigned old_mode_ce0;
|
|
static unsigned old_mode_ce1;
|
|
static unsigned old_mode_sclk;
|
|
static unsigned old_mode_miso;
|
|
static unsigned old_mode_mosi;
|
|
|
|
static uint32_t old_spi_cs;
|
|
static uint32_t old_spi_clk;
|
|
|
|
static unsigned old_mode_ace0;
|
|
static unsigned old_mode_ace1;
|
|
static unsigned old_mode_ace2;
|
|
static unsigned old_mode_asclk;
|
|
static unsigned old_mode_amiso;
|
|
static unsigned old_mode_amosi;
|
|
|
|
static uint32_t old_spi_cntl0;
|
|
static uint32_t old_spi_cntl1;
|
|
|
|
static uint32_t bscFR;
|
|
|
|
/* const --------------------------------------------------------- */
|
|
|
|
static const uint8_t clkDef[PI_MAX_GPIO + 1] =
|
|
{
|
|
/* 0 1 2 3 4 5 6 7 8 9 */
|
|
/* 0 */ 0x00, 0x00, 0x00, 0x00, 0x84, 0x94, 0xA4, 0x00, 0x00, 0x00,
|
|
/* 1 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
/* 2 */ 0x82, 0x92, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
/* 3 */ 0x00, 0x00, 0x84, 0x00, 0x84, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
/* 4 */ 0x00, 0x00, 0x94, 0xA4, 0x94, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
/* 5 */ 0x00, 0x00, 0x00, 0x00,
|
|
};
|
|
|
|
/*
|
|
7 6 5 4 3 2 1 0
|
|
V . C C . M M M
|
|
|
|
V: 0 no clock, 1 has a clock
|
|
CC: 00 CLK0, 01 CLK1, 10 CLK2
|
|
M: 100 ALT0, 010 ALT5
|
|
|
|
gpio4 GPCLK0 ALT0
|
|
gpio5 GPCLK1 ALT0 B+ and compute module only (reserved for system use)
|
|
gpio6 GPCLK2 ALT0 B+ and compute module only
|
|
gpio20 GPCLK0 ALT5 B+ and compute module only
|
|
gpio21 GPCLK1 ALT5 Not available on Rev.2 B (reserved for system use)
|
|
|
|
gpio32 GPCLK0 ALT0 Compute module only
|
|
gpio34 GPCLK0 ALT0 Compute module only
|
|
gpio42 GPCLK1 ALT0 Compute module only (reserved for system use)
|
|
gpio43 GPCLK2 ALT0 Compute module only
|
|
gpio44 GPCLK1 ALT0 Compute module only (reserved for system use)
|
|
*/
|
|
|
|
static const uint8_t PWMDef[PI_MAX_GPIO + 1] =
|
|
{
|
|
/* 0 1 2 3 4 5 6 7 8 9 */
|
|
/* 0 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
/* 1 */ 0x00, 0x00, 0x84, 0x94, 0x00, 0x00, 0x00, 0x00, 0x82, 0x92,
|
|
/* 2 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
/* 3 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
/* 4 */ 0x84, 0x94, 0x00, 0x00, 0x00, 0x94, 0x00, 0x00, 0x00, 0x00,
|
|
/* 5 */ 0x00, 0x00, 0x85, 0x95,
|
|
};
|
|
|
|
/*
|
|
7 6 5 4 3 2 1 0
|
|
V . . P . M M M
|
|
|
|
V: 0 no PWM, 1 has a PWM
|
|
P: 0 PWM0, 1 PWM1
|
|
M: 010 ALT5, 100 ALT0, 101 ALT1
|
|
|
|
gpio12 pwm0 ALT0
|
|
gpio13 pwm1 ALT0
|
|
gpio18 pwm0 ALT5
|
|
gpio19 pwm1 ALT5
|
|
gpio40 pwm0 ALT0
|
|
gpio41 pwm1 ALT0
|
|
gpio45 pwm1 ALT0
|
|
gpio52 pwm0 ALT1
|
|
gpio53 pwm1 ALT1
|
|
*/
|
|
|
|
static const clkCfg_t clkCfg[]=
|
|
{
|
|
/* valid servo */
|
|
{ 0, 0}, /* 0 */
|
|
{ 1, 17}, /* 1 */
|
|
{ 1, 16}, /* 2 */
|
|
{ 0, 0}, /* 3 */
|
|
{ 1, 15}, /* 4 */
|
|
{ 1, 14}, /* 5 */
|
|
{ 0, 0}, /* 6 */
|
|
{ 0, 0}, /* 7 */
|
|
{ 1, 13}, /* 8 */
|
|
{ 0, 0}, /* 9 */
|
|
{ 1, 12}, /* 10 */
|
|
};
|
|
|
|
static const uint16_t pwmCycles[PWM_FREQS]=
|
|
{ 1, 2, 4, 5, 8, 10, 16, 20, 25,
|
|
32, 40, 50, 80, 100, 160, 200, 400, 800};
|
|
|
|
static const uint16_t pwmRealRange[PWM_FREQS]=
|
|
{ 25, 50, 100, 125, 200, 250, 400, 500, 625,
|
|
800, 1000, 1250, 2000, 2500, 4000, 5000, 10000, 20000};
|
|
|
|
/* prototype ----------------------------------------------------- */
|
|
|
|
static void intNotifyBits(void);
|
|
|
|
static void intScriptBits(void);
|
|
|
|
static void intScriptEventBits(void);
|
|
|
|
static int gpioNotifyOpenInBand(int fd);
|
|
|
|
static void initHWClk
|
|
(int clkCtl, int clkDiv, int clkSrc, int divI, int divF, int MASH);
|
|
|
|
static void initDMAgo(volatile uint32_t *dmaAddr, uint32_t cbAddr);
|
|
|
|
int gpioWaveTxStart(unsigned wave_mode); /* deprecated */
|
|
|
|
static void closeOrphanedNotifications(int slot, int fd);
|
|
|
|
|
|
/* ======================================================================= */
|
|
|
|
int myScriptNameValid(char *name)
|
|
{
|
|
int i, c, len, valid;
|
|
|
|
len = strlen(name);
|
|
|
|
valid = 1;
|
|
|
|
for (i=0; i<len; i++)
|
|
{
|
|
c = name[i];
|
|
|
|
if ((!isalnum(c)) && (c != '_') && (c != '-'))
|
|
{
|
|
valid = 0;
|
|
break;
|
|
}
|
|
}
|
|
return valid;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static char * myTimeStamp()
|
|
{
|
|
static struct timeval last;
|
|
static char buf[32];
|
|
struct timeval now;
|
|
|
|
struct tm tmp;
|
|
|
|
gettimeofday(&now, NULL);
|
|
|
|
if (now.tv_sec != last.tv_sec)
|
|
{
|
|
localtime_r(&now.tv_sec, &tmp);
|
|
strftime(buf, sizeof(buf), "%F %T", &tmp);
|
|
last.tv_sec = now.tv_sec;
|
|
}
|
|
|
|
return buf;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int myPathBad(char *name)
|
|
{
|
|
int i, c, len, in_part, parts, last_char_dot;
|
|
char *bad="/*?.";
|
|
|
|
parts = 0;
|
|
in_part = 0;
|
|
last_char_dot = 0;
|
|
|
|
if (strstr(name, "..")) return 1;
|
|
if (strstr(name, "\\.")) return 1;
|
|
|
|
len = strlen(name);
|
|
|
|
for (i=0; i<len; i++)
|
|
{
|
|
c = name[i];
|
|
|
|
if (memchr(bad, c, 4)) /* wildcard or directory character */
|
|
{
|
|
if (c == '.')
|
|
{
|
|
if (last_char_dot) return 1;
|
|
last_char_dot = 1;
|
|
}
|
|
else last_char_dot = 0;
|
|
|
|
in_part = 0;
|
|
}
|
|
else /* normal character */
|
|
{
|
|
last_char_dot = 0;
|
|
|
|
if (!in_part) parts++;
|
|
|
|
in_part = 1;
|
|
}
|
|
}
|
|
|
|
if (parts < 2) return 1; else return 0;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static char *myBuf2Str(unsigned count, char *buf)
|
|
{
|
|
static char str[128];
|
|
int i, c;
|
|
|
|
if (count && buf)
|
|
{
|
|
if (count > 40) c = 40; else c = count;
|
|
|
|
for (i=0; i<c; i++) sprintf(str+(3*i), "%02X ", buf[i]);
|
|
str[(3*c)-1] = 0;
|
|
}
|
|
else str[0] = 0;
|
|
|
|
return str;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static int my_smbus_access(
|
|
int fd, char rw, uint8_t cmd, int size, union my_smbus_data *data)
|
|
{
|
|
struct my_smbus_ioctl_data args;
|
|
|
|
DBG(DBG_INTERNAL, "rw=%d reg=%d cmd=%d data=%s",
|
|
rw, cmd, size, myBuf2Str(data->byte+1, (char*)data));
|
|
|
|
args.read_write = rw;
|
|
args.command = cmd;
|
|
args.size = size;
|
|
args.data = data;
|
|
|
|
return ioctl(fd, PI_I2C_SMBUS, &args);
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static void myGpioSetMode(unsigned gpio, unsigned mode)
|
|
{
|
|
int reg, shift;
|
|
|
|
reg = gpio/10;
|
|
shift = (gpio%10) * 3;
|
|
|
|
gpioReg[reg] = (gpioReg[reg] & ~(7<<shift)) | (mode<<shift);
|
|
}
|
|
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static int myGpioRead(unsigned gpio)
|
|
{
|
|
if ((*(gpioReg + GPLEV0 + BANK) & BIT) != 0) return PI_ON;
|
|
else return PI_OFF;
|
|
}
|
|
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static void myGpioWrite(unsigned gpio, unsigned level)
|
|
{
|
|
if (level == PI_OFF) *(gpioReg + GPCLR0 + BANK) = BIT;
|
|
else *(gpioReg + GPSET0 + BANK) = BIT;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static void myGpioSleep(int seconds, int micros)
|
|
{
|
|
struct timespec ts, rem;
|
|
|
|
ts.tv_sec = seconds;
|
|
ts.tv_nsec = micros * 1000;
|
|
|
|
while (clock_nanosleep(CLOCK_REALTIME, 0, &ts, &rem))
|
|
{
|
|
/* copy remaining time to ts */
|
|
ts = rem;
|
|
}
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static uint32_t myGpioDelay(uint32_t micros)
|
|
{
|
|
uint32_t start;
|
|
|
|
start = systReg[SYST_CLO];
|
|
|
|
if (micros <= PI_MAX_BUSY_DELAY)
|
|
{
|
|
while ((systReg[SYST_CLO] - start) <= micros);
|
|
}
|
|
else
|
|
{
|
|
myGpioSleep(micros/MILLION, micros%MILLION);
|
|
}
|
|
|
|
return (systReg[SYST_CLO] - start);
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static void myCreatePipe(char * name, int perm)
|
|
{
|
|
unlink(name);
|
|
|
|
mkfifo(name, perm);
|
|
|
|
if (chmod(name, perm) < 0)
|
|
{
|
|
DBG(DBG_ALWAYS, "Can't set permissions (%d) for %s, %m", perm, name);
|
|
return;
|
|
}
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static void myOffPageSlot(int pos, int * page, int * slot)
|
|
{
|
|
*page = pos/OFF_PER_IPAGE;
|
|
*slot = pos%OFF_PER_IPAGE;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static void myLvsPageSlot(int pos, int * page, int * slot)
|
|
{
|
|
*page = pos/LVS_PER_IPAGE;
|
|
*slot = pos%LVS_PER_IPAGE;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static void myTckPageSlot(int pos, int * page, int * slot)
|
|
{
|
|
*page = pos/TCK_PER_IPAGE;
|
|
*slot = pos%TCK_PER_IPAGE;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static uint32_t myGetLevel(int pos)
|
|
{
|
|
uint32_t level;
|
|
int page, slot;
|
|
|
|
myLvsPageSlot(pos, &page, &slot);
|
|
|
|
level = dmaIVirt[page]->level[slot];
|
|
|
|
return level;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static int myI2CGetPar(char *inBuf, int *inPos, int inLen, int *esc)
|
|
{
|
|
int bytes;
|
|
|
|
if (*esc) bytes = 2; else bytes = 1;
|
|
|
|
*esc = 0;
|
|
|
|
if (*inPos <= (inLen - bytes))
|
|
{
|
|
if (bytes == 1)
|
|
{
|
|
return inBuf[(*inPos)++];
|
|
}
|
|
else
|
|
{
|
|
(*inPos) += 2;
|
|
return inBuf[*inPos-2] + (inBuf[*inPos-1]<<8);
|
|
}
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static uint32_t myGetTick(int pos)
|
|
{
|
|
uint32_t tick;
|
|
int page, slot;
|
|
|
|
myTckPageSlot(pos, &page, &slot);
|
|
|
|
tick = dmaIVirt[page]->tick[slot];
|
|
|
|
return tick;
|
|
}
|
|
|
|
static int myPermit(unsigned gpio)
|
|
{
|
|
if (gpio <= PI_MAX_GPIO)
|
|
{
|
|
if (gpioMask & ((uint64_t)(1)<<gpio)) return 1;
|
|
else return 0;
|
|
}
|
|
return 1; /* will fail for bad gpio number */
|
|
}
|
|
|
|
static void flushMemory(void)
|
|
{
|
|
static int val = 0;
|
|
|
|
void *dummy;
|
|
|
|
dummy = mmap(
|
|
0, (FLUSH_PAGES*PAGE_SIZE),
|
|
PROT_READ|PROT_WRITE|PROT_EXEC,
|
|
MAP_SHARED|MAP_ANONYMOUS|MAP_NORESERVE|MAP_LOCKED,
|
|
-1, 0);
|
|
|
|
if (dummy == MAP_FAILED)
|
|
{
|
|
DBG(DBG_STARTUP, "mmap dummy failed (%m)");
|
|
}
|
|
else
|
|
{
|
|
memset(dummy, val++, (FLUSH_PAGES*PAGE_SIZE));
|
|
memset(dummy, val++, (FLUSH_PAGES*PAGE_SIZE));
|
|
munmap(dummy, FLUSH_PAGES*PAGE_SIZE);
|
|
}
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static void wfRx_lock(int i)
|
|
{
|
|
pthread_mutex_lock(&wfRx[i].mutex);
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static void wfRx_unlock(int i)
|
|
{
|
|
pthread_mutex_unlock(&wfRx[i].mutex);
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static void spinWhileStarting(void)
|
|
{
|
|
while (runState == PI_STARTING)
|
|
{
|
|
if (piCores == 1) myGpioDelay(1000);
|
|
else flushMemory();
|
|
}
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static int myDoCommand(uintptr_t *p, unsigned bufSize, char *buf)
|
|
{
|
|
int res, i, j;
|
|
uint32_t mask;
|
|
uint32_t tmp1, tmp2, tmp3, tmp4, tmp5;
|
|
gpioPulse_t *pulse;
|
|
bsc_xfer_t xfer;
|
|
int masked;
|
|
res = 0;
|
|
|
|
switch (p[0])
|
|
{
|
|
case PI_CMD_BC1:
|
|
mask = gpioMask;
|
|
|
|
res = gpioWrite_Bits_0_31_Clear(p[1]&mask);
|
|
|
|
if ((mask | p[1]) != mask)
|
|
{
|
|
DBG(DBG_USER,
|
|
"gpioWrite_Bits_0_31_Clear: bad bits %08"PRIXPTR" (permissions %08X)",
|
|
p[1], mask);
|
|
res = PI_SOME_PERMITTED;
|
|
}
|
|
break;
|
|
|
|
case PI_CMD_BC2:
|
|
mask = gpioMask>>32;
|
|
|
|
res = gpioWrite_Bits_32_53_Clear(p[1]&mask);
|
|
|
|
if ((mask | p[1]) != mask)
|
|
{
|
|
DBG(DBG_USER,
|
|
"gpioWrite_Bits_32_53_Clear: bad bits %08"PRIXPTR" (permissions %08X)",
|
|
p[1], mask);
|
|
res = PI_SOME_PERMITTED;
|
|
}
|
|
break;
|
|
|
|
case PI_CMD_BI2CC:
|
|
res = bbI2CClose(p[1]);
|
|
break;
|
|
|
|
case PI_CMD_BI2CO:
|
|
memcpy(&p[4], buf, 4);
|
|
res = bbI2COpen(p[1], p[2], p[4]);
|
|
break;
|
|
|
|
case PI_CMD_BI2CZ:
|
|
/* use half buffer for write, half buffer for read */
|
|
if (p[3] > (bufSize/2)) p[3] = bufSize/2;
|
|
res = bbI2CZip(p[1], buf, p[3], buf+(bufSize/2), bufSize/2);
|
|
if (res > 0)
|
|
{
|
|
memcpy(buf, buf+(bufSize/2), res);
|
|
}
|
|
break;
|
|
|
|
case PI_CMD_BSCX:
|
|
xfer.control = p[1];
|
|
if (p[3] > BSC_FIFO_SIZE) p[3] = BSC_FIFO_SIZE;
|
|
xfer.txCnt = p[3];
|
|
if (p[3]) memcpy(&xfer.txBuf, buf, p[3]);
|
|
res = bscXfer(&xfer);
|
|
if (res >= 0)
|
|
{
|
|
memcpy(buf, &res, 4);
|
|
res = 4 + xfer.rxCnt;
|
|
if (res > 4) memcpy(buf+4, &xfer.rxBuf, res-4);
|
|
}
|
|
break;
|
|
|
|
case PI_CMD_BSPIO:
|
|
|
|
memcpy(&tmp1, buf+ 0, 4); // MISO
|
|
memcpy(&tmp2, buf+ 4, 4); // MOSI
|
|
memcpy(&tmp3, buf+ 8, 4); // SCLK
|
|
memcpy(&tmp4, buf+12, 4); // baud
|
|
memcpy(&tmp5, buf+16, 4); // flags
|
|
|
|
if (!myPermit(p[1]))
|
|
{
|
|
DBG(DBG_USER,
|
|
"bbSPIOpen: gpio %"PRIdPTR", no permission to update CS", p[1]);
|
|
res = PI_NOT_PERMITTED;
|
|
}
|
|
|
|
if (!myPermit(tmp1))
|
|
{
|
|
DBG(DBG_USER,
|
|
"bbSPIOpen: gpio %d, no permission to update MISO", tmp1);
|
|
res = PI_NOT_PERMITTED;
|
|
}
|
|
|
|
if (!myPermit(tmp2))
|
|
{
|
|
DBG(DBG_USER,
|
|
"bbSPIOpen: gpio %d, no permission to update MOSI", tmp2);
|
|
res = PI_NOT_PERMITTED;
|
|
}
|
|
|
|
if (!myPermit(tmp3))
|
|
{
|
|
DBG(DBG_USER,
|
|
"bbSPIOpen: gpio %d, no permission to update SCLK", tmp3);
|
|
res = PI_NOT_PERMITTED;
|
|
}
|
|
|
|
if (!res) res = bbSPIOpen(p[1], tmp1, tmp2, tmp3, tmp4, tmp5);
|
|
break;
|
|
|
|
case PI_CMD_BSPIC:
|
|
res = bbSPIClose(p[1]);
|
|
break;
|
|
|
|
case PI_CMD_BSPIX:
|
|
if (p[3] > bufSize) p[3] = bufSize;
|
|
res = bbSPIXfer(p[1], buf, buf, p[3]);
|
|
break;
|
|
|
|
case PI_CMD_BR1: res = gpioRead_Bits_0_31(); break;
|
|
|
|
case PI_CMD_BR2: res = gpioRead_Bits_32_53(); break;
|
|
|
|
case PI_CMD_BS1:
|
|
mask = gpioMask;
|
|
|
|
res = gpioWrite_Bits_0_31_Set(p[1]&mask);
|
|
|
|
if ((mask | p[1]) != mask)
|
|
{
|
|
DBG(DBG_USER,
|
|
"gpioWrite_Bits_0_31_Set: bad bits %08"PRIXPTR" (permissions %08X)",
|
|
p[1], mask);
|
|
res = PI_SOME_PERMITTED;
|
|
}
|
|
break;
|
|
|
|
case PI_CMD_BS2:
|
|
mask = gpioMask>>32;
|
|
|
|
res = gpioWrite_Bits_32_53_Set(p[1]&mask);
|
|
|
|
if ((mask | p[1]) != mask)
|
|
{
|
|
DBG(DBG_USER,
|
|
"gpioWrite_Bits_32_53_Set: bad bits %08"PRIXPTR" (permissions %08X)",
|
|
p[1], mask);
|
|
res = PI_SOME_PERMITTED;
|
|
}
|
|
break;
|
|
|
|
case PI_CMD_CF1:
|
|
res = gpioCustom1(p[1], p[2], buf, p[3]);
|
|
break;
|
|
|
|
case PI_CMD_CF2:
|
|
/* a couple of extra precautions for untrusted code */
|
|
if (p[2] > bufSize) p[2] = bufSize;
|
|
res = gpioCustom2(p[1], buf, p[3], buf, p[2]);
|
|
if (res > p[2]) res = p[2];
|
|
break;
|
|
|
|
case PI_CMD_CGI: res = gpioCfgGetInternals(); break;
|
|
|
|
case PI_CMD_CSI: res = gpioCfgSetInternals(p[1]); break;
|
|
|
|
case PI_CMD_EVM: res = eventMonitor(p[1], p[2]); break;
|
|
|
|
case PI_CMD_EVT: res = eventTrigger(p[1]); break;
|
|
|
|
case PI_CMD_FC: res = fileClose(p[1]); break;
|
|
|
|
case PI_CMD_FG:
|
|
res = gpioGlitchFilter(p[1], p[2]);
|
|
break;
|
|
|
|
case PI_CMD_FL:
|
|
if (p[1] > bufSize) p[1] = bufSize;
|
|
res = fileList(buf, buf, p[1]);
|
|
break;
|
|
|
|
case PI_CMD_FN:
|
|
memcpy(&p[4], buf, 4);
|
|
res = gpioNoiseFilter(p[1], p[2], p[4]);
|
|
break;
|
|
|
|
case PI_CMD_FO: res = fileOpen(buf, p[1]); break;
|
|
|
|
case PI_CMD_FR:
|
|
if (p[2] > bufSize) p[2] = bufSize;
|
|
res = fileRead(p[1], buf, p[2]);
|
|
break;
|
|
|
|
case PI_CMD_FS:
|
|
memcpy(&p[4], buf, 4);
|
|
res = fileSeek(p[1], p[2], p[4]);
|
|
break;
|
|
|
|
case PI_CMD_FW: res = fileWrite(p[1], buf, p[3]); break;
|
|
|
|
case PI_CMD_GDC: res = gpioGetPWMdutycycle(p[1]); break;
|
|
|
|
case PI_CMD_GPW: res = gpioGetServoPulsewidth(p[1]); break;
|
|
|
|
case PI_CMD_HC:
|
|
/* special case to allow password in upper byte */
|
|
if (myPermit(p[1]&0xFFFFFF)) res = gpioHardwareClock(p[1], p[2]);
|
|
else
|
|
{
|
|
DBG(DBG_USER,
|
|
"gpioHardwareClock: gpio %"PRIdPTR", no permission to update",
|
|
p[1] & 0xFFFFFF);
|
|
res = PI_NOT_PERMITTED;
|
|
}
|
|
break;
|
|
|
|
case PI_CMD_HELP: break;
|
|
|
|
case PI_CMD_HP:
|
|
if (myPermit(p[1]))
|
|
{
|
|
memcpy(&p[4], buf, 4);
|
|
res = gpioHardwarePWM(p[1], p[2], p[4]);
|
|
}
|
|
else
|
|
{
|
|
DBG(DBG_USER,
|
|
"gpioHardwarePWM: gpio %"PRIdPTR", no permission to update", p[1]);
|
|
res = PI_NOT_PERMITTED;
|
|
}
|
|
break;
|
|
|
|
case PI_CMD_HWVER: res = gpioHardwareRevision(); break;
|
|
|
|
|
|
|
|
case PI_CMD_I2CC: res = i2cClose(p[1]); break;
|
|
|
|
case PI_CMD_I2CO:
|
|
memcpy(&p[4], buf, 4);
|
|
res = i2cOpen(p[1], p[2], p[4]);
|
|
break;
|
|
|
|
case PI_CMD_I2CPC:
|
|
memcpy(&p[4], buf, 4);
|
|
res = i2cProcessCall(p[1], p[2], p[4]);
|
|
break;
|
|
|
|
case PI_CMD_I2CPK:
|
|
res = i2cBlockProcessCall(p[1], p[2], buf, p[3]);
|
|
break;
|
|
|
|
case PI_CMD_I2CRB: res = i2cReadByteData(p[1], p[2]); break;
|
|
|
|
case PI_CMD_I2CRD:
|
|
if (p[2] > bufSize) p[2] = bufSize;
|
|
res = i2cReadDevice(p[1], buf, p[2]);
|
|
break;
|
|
|
|
case PI_CMD_I2CRI:
|
|
memcpy(&p[4], buf, 4);
|
|
res = i2cReadI2CBlockData(p[1], p[2], buf, p[4]);
|
|
break;
|
|
|
|
case PI_CMD_I2CRK:
|
|
res = i2cReadBlockData(p[1], p[2], buf);
|
|
break;
|
|
|
|
case PI_CMD_I2CRS: res = i2cReadByte(p[1]); break;
|
|
|
|
case PI_CMD_I2CRW: res = i2cReadWordData(p[1], p[2]); break;
|
|
|
|
case PI_CMD_I2CWB:
|
|
memcpy(&p[4], buf, 4);
|
|
res = i2cWriteByteData(p[1], p[2], p[4]);
|
|
break;
|
|
|
|
case PI_CMD_I2CWD:
|
|
res = i2cWriteDevice(p[1], buf, p[3]);
|
|
break;
|
|
|
|
case PI_CMD_I2CWI:
|
|
res = i2cWriteI2CBlockData(p[1], p[2], buf, p[3]);
|
|
break;
|
|
|
|
case PI_CMD_I2CWK:
|
|
res = i2cWriteBlockData(p[1], p[2], buf, p[3]);
|
|
break;
|
|
|
|
case PI_CMD_I2CWQ: res = i2cWriteQuick(p[1], p[2]); break;
|
|
|
|
case PI_CMD_I2CWS: res = i2cWriteByte(p[1], p[2]); break;
|
|
|
|
case PI_CMD_I2CWW:
|
|
memcpy(&p[4], buf, 4);
|
|
res = i2cWriteWordData(p[1], p[2], p[4]);
|
|
break;
|
|
|
|
case PI_CMD_I2CZ:
|
|
/* use half buffer for write, half buffer for read */
|
|
if (p[3] > (bufSize/2)) p[3] = bufSize/2;
|
|
res = i2cZip(p[1], buf, p[3], buf+(bufSize/2), bufSize/2);
|
|
if (res > 0)
|
|
{
|
|
memcpy(buf, buf+(bufSize/2), res);
|
|
}
|
|
break;
|
|
|
|
case PI_CMD_MICS:
|
|
if (p[1] <= PI_MAX_MICS_DELAY) myGpioDelay(p[1]);
|
|
else res = PI_BAD_MICS_DELAY;
|
|
break;
|
|
|
|
case PI_CMD_MILS:
|
|
if (p[1] <= PI_MAX_MILS_DELAY) myGpioDelay(p[1] * 1000);
|
|
else res = PI_BAD_MILS_DELAY;
|
|
break;
|
|
|
|
case PI_CMD_MODEG: res = gpioGetMode(p[1]); break;
|
|
|
|
case PI_CMD_MODES:
|
|
if (myPermit(p[1])) res = gpioSetMode(p[1], p[2]);
|
|
else
|
|
{
|
|
DBG(DBG_USER,
|
|
"gpioSetMode: gpio %"PRIdPTR", no permission to update", p[1]);
|
|
res = PI_NOT_PERMITTED;
|
|
}
|
|
break;
|
|
|
|
case PI_CMD_NB: res = gpioNotifyBegin(p[1], p[2]); break;
|
|
|
|
case PI_CMD_NC: res = gpioNotifyClose(p[1]); break;
|
|
|
|
case PI_CMD_NO: res = gpioNotifyOpen(); break;
|
|
|
|
case PI_CMD_NP: res = gpioNotifyPause(p[1]); break;
|
|
|
|
case PI_CMD_PADG: res = gpioGetPad(p[1]); break;
|
|
|
|
case PI_CMD_PADS: res = gpioSetPad(p[1], p[2]); break;
|
|
|
|
case PI_CMD_PFG: res = gpioGetPWMfrequency(p[1]); break;
|
|
|
|
case PI_CMD_PFS:
|
|
if (myPermit(p[1])) res = gpioSetPWMfrequency(p[1], p[2]);
|
|
else
|
|
{
|
|
DBG(DBG_USER,
|
|
"gpioSetPWMfrequency: gpio %"PRIdPTR", no permission to update", p[1]);
|
|
res = PI_NOT_PERMITTED;
|
|
}
|
|
break;
|
|
|
|
case PI_CMD_PIGPV: res = gpioVersion(); break;
|
|
|
|
case PI_CMD_PRG: res = gpioGetPWMrange(p[1]); break;
|
|
|
|
case PI_CMD_PROC:
|
|
res = gpioStoreScript(buf);
|
|
break;
|
|
|
|
case PI_CMD_PROCD: res = gpioDeleteScript(p[1]); break;
|
|
|
|
case PI_CMD_PROCP:
|
|
res = gpioScriptStatus(p[1], (uint32_t *)buf);
|
|
break;
|
|
|
|
case PI_CMD_PROCR:
|
|
res = gpioRunScript(p[1], p[3]/4, (uint32_t *)buf);
|
|
break;
|
|
|
|
case PI_CMD_PROCS: res = gpioStopScript(p[1]); break;
|
|
|
|
case PI_CMD_PROCU:
|
|
res = gpioUpdateScript(p[1], p[3]/4, (uint32_t *)buf);
|
|
break;
|
|
|
|
case PI_CMD_PRRG: res = gpioGetPWMrealRange(p[1]); break;
|
|
|
|
case PI_CMD_PRS:
|
|
if (myPermit(p[1])) res = gpioSetPWMrange(p[1], p[2]);
|
|
else
|
|
{
|
|
DBG(DBG_USER,
|
|
"gpioSetPWMrange: gpio %"PRIdPTR", no permission to update", p[1]);
|
|
res = PI_NOT_PERMITTED;
|
|
}
|
|
break;
|
|
|
|
case PI_CMD_PUD:
|
|
if (myPermit(p[1])) res = gpioSetPullUpDown(p[1], p[2]);
|
|
else
|
|
{
|
|
DBG(DBG_USER,
|
|
"gpioSetPullUpDown: gpio %"PRIdPTR", no permission to update", p[1]);
|
|
res = PI_NOT_PERMITTED;
|
|
}
|
|
break;
|
|
|
|
case PI_CMD_PWM:
|
|
if (myPermit(p[1])) res = gpioPWM(p[1], p[2]);
|
|
else
|
|
{
|
|
DBG(DBG_USER, "gpioPWM: gpio %"PRIdPTR", no permission to update", p[1]);
|
|
res = PI_NOT_PERMITTED;
|
|
}
|
|
break;
|
|
|
|
case PI_CMD_READ: res = gpioRead(p[1]); break;
|
|
|
|
case PI_CMD_SERVO:
|
|
if (myPermit(p[1])) res = gpioServo(p[1], p[2]);
|
|
else
|
|
{
|
|
DBG(DBG_USER,
|
|
"gpioServo: gpio %"PRIdPTR", no permission to update", p[1]);
|
|
res = PI_NOT_PERMITTED;
|
|
}
|
|
break;
|
|
|
|
|
|
|
|
case PI_CMD_SERRB: res = serReadByte(p[1]); break;
|
|
|
|
case PI_CMD_SERWB: res = serWriteByte(p[1], p[2]); break;
|
|
|
|
case PI_CMD_SERC: res = serClose(p[1]); break;
|
|
|
|
case PI_CMD_SERDA: res = serDataAvailable(p[1]); break;
|
|
|
|
case PI_CMD_SERO: res = serOpen(buf, p[1], p[2]); break;
|
|
|
|
case PI_CMD_SERR:
|
|
if (p[2] > bufSize) p[2] = bufSize;
|
|
res = serRead(p[1], buf, p[2]);
|
|
break;
|
|
|
|
case PI_CMD_SERW: res = serWrite(p[1], buf, p[3]); break;
|
|
|
|
|
|
case PI_CMD_SHELL:
|
|
res = shell(buf, buf+p[1]+1);
|
|
break;
|
|
|
|
|
|
case PI_CMD_SLR:
|
|
if (p[2] > bufSize) p[2] = bufSize;
|
|
res = gpioSerialRead(p[1], buf, p[2]);
|
|
break;
|
|
|
|
case PI_CMD_SLRC: res = gpioSerialReadClose(p[1]); break;
|
|
|
|
case PI_CMD_SLRO:
|
|
memcpy(&p[4], buf, 4);
|
|
res = gpioSerialReadOpen(p[1], p[2], p[4]); break;
|
|
|
|
case PI_CMD_SLRI: res = gpioSerialReadInvert(p[1], p[2]); break;
|
|
|
|
case PI_CMD_SPIC:
|
|
res = spiClose(p[1]);
|
|
break;
|
|
|
|
case PI_CMD_SPIO:
|
|
memcpy(&p[4], buf, 4);
|
|
res = spiOpen(p[1], p[2], p[4]);
|
|
break;
|
|
|
|
case PI_CMD_SPIR:
|
|
if (p[2] > bufSize) p[2] = bufSize;
|
|
res = spiRead(p[1], buf, p[2]);
|
|
break;
|
|
|
|
case PI_CMD_SPIW:
|
|
if (p[3] > bufSize) p[3] = bufSize;
|
|
res = spiWrite(p[1], buf, p[3]);
|
|
break;
|
|
|
|
case PI_CMD_SPIX:
|
|
if (p[3] > bufSize) p[3] = bufSize;
|
|
res = spiXfer(p[1], buf, buf, p[3]);
|
|
break;
|
|
|
|
case PI_CMD_TICK: res = gpioTick(); break;
|
|
|
|
case PI_CMD_TRIG:
|
|
if (myPermit(p[1]))
|
|
{
|
|
memcpy(&p[4], buf, 4);
|
|
res = gpioTrigger(p[1], p[2], p[4]);
|
|
}
|
|
else
|
|
{
|
|
DBG(DBG_USER,
|
|
"gpioTrigger: gpio %"PRIdPTR", no permission to update", p[1]);
|
|
res = PI_NOT_PERMITTED;
|
|
}
|
|
break;
|
|
|
|
case PI_CMD_WDOG: res = gpioSetWatchdog(p[1], p[2]); break;
|
|
|
|
case PI_CMD_WRITE:
|
|
if (myPermit(p[1])) res = gpioWrite(p[1], p[2]);
|
|
else
|
|
{
|
|
DBG(DBG_USER, "gpioWrite: gpio %"PRIdPTR", no permission to update", p[1]);
|
|
res = PI_NOT_PERMITTED;
|
|
}
|
|
break;
|
|
|
|
|
|
|
|
case PI_CMD_WVAG:
|
|
|
|
/* need to mask off any non permitted gpios */
|
|
|
|
mask = gpioMask;
|
|
pulse = (gpioPulse_t *)buf;
|
|
j = p[3]/sizeof(gpioPulse_t);
|
|
masked = 0;
|
|
|
|
for (i=0; i<j; i++)
|
|
{
|
|
tmp1 = pulse[i].gpioOn & mask;
|
|
if (tmp1 != pulse[i].gpioOn)
|
|
{
|
|
pulse[i].gpioOn = tmp1;
|
|
masked = 1;
|
|
}
|
|
|
|
tmp1 = pulse[i].gpioOff & mask;
|
|
if (tmp1 != pulse[i].gpioOff)
|
|
{
|
|
pulse[i].gpioOff = tmp1;
|
|
masked = 1;
|
|
}
|
|
DBG(DBG_SCRIPT, "on=%X off=%X delay=%d",
|
|
pulse[i].gpioOn, pulse[i].gpioOff, pulse[i].usDelay);
|
|
}
|
|
|
|
res = gpioWaveAddGeneric(j, pulse);
|
|
|
|
/* report permission error unless another error occurred */
|
|
if (masked && (res >= 0)) res = PI_SOME_PERMITTED;
|
|
|
|
break;
|
|
|
|
case PI_CMD_WVAS:
|
|
if (myPermit(p[1]))
|
|
{
|
|
memcpy(&tmp1, buf, 4); /* databits */
|
|
memcpy(&tmp2, buf+4, 4); /* stophalfbits */
|
|
memcpy(&tmp3, buf+8, 4); /* offset */
|
|
res = gpioWaveAddSerial
|
|
(p[1], p[2], tmp1, tmp2, tmp3, p[3]-12, buf+12);
|
|
}
|
|
else
|
|
{
|
|
DBG(
|
|
DBG_USER,
|
|
"gpioWaveAddSerial: gpio %"PRIdPTR", no permission to update", p[1]);
|
|
res = PI_NOT_PERMITTED;
|
|
}
|
|
break;
|
|
|
|
case PI_CMD_WVBSY: res = gpioWaveTxBusy(); break;
|
|
|
|
case PI_CMD_WVCHA:
|
|
if (p[3] > bufSize) p[3] = bufSize;
|
|
res = gpioWaveChain(buf, p[3]);
|
|
break;
|
|
|
|
|
|
case PI_CMD_WVCLR: res = gpioWaveClear(); break;
|
|
|
|
case PI_CMD_WVCRE: res = gpioWaveCreate(); break;
|
|
|
|
case PI_CMD_WVCAP:
|
|
/* Make WVCAP variadic */
|
|
if (p[3] == 4)
|
|
{
|
|
memcpy(&tmp3, buf, 4); /* percent TOOL */
|
|
res = gpioWaveCreatePad(p[1], p[2], tmp3); /* rawWaveAdd* usage */
|
|
break;
|
|
}
|
|
if (p[2] && p[3]==0)
|
|
{
|
|
res = gpioWaveCreatePad(p[1], p[2], 0);
|
|
break;
|
|
}
|
|
if (p[2]==0 && p[3]==0)
|
|
{
|
|
res = gpioWaveCreatePad(p[1], p[1], 0); /* typical usage */
|
|
break;
|
|
}
|
|
res = PI_BAD_WAVE_ID; // FIX?
|
|
break;
|
|
|
|
case PI_CMD_WVDEL: res = gpioWaveDelete(p[1]); break;
|
|
|
|
case PI_CMD_WVGO: res = gpioWaveTxStart(PI_WAVE_MODE_ONE_SHOT); break;
|
|
|
|
case PI_CMD_WVGOR: res = gpioWaveTxStart(PI_WAVE_MODE_REPEAT); break;
|
|
|
|
case PI_CMD_WVHLT: res = gpioWaveTxStop(); break;
|
|
|
|
case PI_CMD_WVNEW: res = gpioWaveAddNew(); break;
|
|
|
|
case PI_CMD_WVSC:
|
|
switch(p[1])
|
|
{
|
|
case 0: res = gpioWaveGetCbs(); break;
|
|
case 1: res = gpioWaveGetHighCbs(); break;
|
|
case 2: res = gpioWaveGetMaxCbs(); break;
|
|
default: res = PI_BAD_WVSC_COMMND;
|
|
}
|
|
break;
|
|
|
|
case PI_CMD_WVSM:
|
|
switch(p[1])
|
|
{
|
|
case 0: res = gpioWaveGetMicros(); break;
|
|
case 1: res = gpioWaveGetHighMicros(); break;
|
|
case 2: res = gpioWaveGetMaxMicros(); break;
|
|
default: res = PI_BAD_WVSM_COMMND;
|
|
}
|
|
break;
|
|
|
|
case PI_CMD_WVSP:
|
|
switch(p[1])
|
|
{
|
|
case 0: res = gpioWaveGetPulses(); break;
|
|
case 1: res = gpioWaveGetHighPulses(); break;
|
|
case 2: res = gpioWaveGetMaxPulses(); break;
|
|
default: res = PI_BAD_WVSP_COMMND;
|
|
}
|
|
break;
|
|
|
|
case PI_CMD_WVTAT: res = gpioWaveTxAt(); break;
|
|
|
|
case PI_CMD_WVTX:
|
|
res = gpioWaveTxSend(p[1], PI_WAVE_MODE_ONE_SHOT); break;
|
|
|
|
case PI_CMD_WVTXM:
|
|
res = gpioWaveTxSend(p[1], p[2]); break;
|
|
|
|
case PI_CMD_WVTXR:
|
|
res = gpioWaveTxSend(p[1], PI_WAVE_MODE_REPEAT); break;
|
|
|
|
default:
|
|
res = PI_UNKNOWN_COMMAND;
|
|
break;
|
|
}
|
|
|
|
return res;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static void mySetGpioOff(unsigned gpio, int pos)
|
|
{
|
|
int page, slot;
|
|
|
|
myOffPageSlot(pos, &page, &slot);
|
|
|
|
dmaIVirt[page]->gpioOff[slot] |= (1<<gpio);
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static void myClearGpioOff(unsigned gpio, int pos)
|
|
{
|
|
int page, slot;
|
|
|
|
myOffPageSlot(pos, &page, &slot);
|
|
|
|
dmaIVirt[page]->gpioOff[slot] &= ~(1<<gpio);
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static void mySetGpioOn(unsigned gpio, int pos)
|
|
{
|
|
int page, slot;
|
|
|
|
page = pos/ON_PER_IPAGE;
|
|
slot = pos%ON_PER_IPAGE;
|
|
|
|
dmaIVirt[page]->gpioOn[slot] |= (1<<gpio);
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static void myClearGpioOn(unsigned gpio, int pos)
|
|
{
|
|
int page, slot;
|
|
|
|
page = pos/ON_PER_IPAGE;
|
|
slot = pos%ON_PER_IPAGE;
|
|
|
|
dmaIVirt[page]->gpioOn[slot] &= ~(1<<gpio);
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static void myGpioSetPwm(unsigned gpio, int oldVal, int newVal)
|
|
{
|
|
int switchGpioOff;
|
|
int newOff, oldOff, realRange, cycles, i;
|
|
int deferOff, deferRng;
|
|
|
|
DBG(DBG_INTERNAL,
|
|
"myGpioSetPwm %d from %d to %d", gpio, oldVal, newVal);
|
|
|
|
switchGpioOff = 0;
|
|
|
|
realRange = pwmRealRange[gpioInfo[gpio].freqIdx];
|
|
|
|
cycles = pwmCycles [gpioInfo[gpio].freqIdx];
|
|
|
|
newOff = (newVal * realRange)/gpioInfo[gpio].range;
|
|
oldOff = (oldVal * realRange)/gpioInfo[gpio].range;
|
|
|
|
deferOff = gpioInfo[gpio].deferOff;
|
|
deferRng = gpioInfo[gpio].deferRng;
|
|
|
|
if (gpioInfo[gpio].deferOff)
|
|
{
|
|
for (i=0; i<SUPERLEVEL; i+=deferRng)
|
|
{
|
|
myClearGpioOff(gpio, i+deferOff);
|
|
}
|
|
gpioInfo[gpio].deferOff = 0;
|
|
}
|
|
|
|
if (newOff != oldOff)
|
|
{
|
|
if (newOff && oldOff) /* PWM CHANGE */
|
|
{
|
|
if (newOff != realRange)
|
|
{
|
|
for (i=0; i<SUPERLEVEL; i+=realRange) mySetGpioOff(gpio, i+newOff);
|
|
}
|
|
|
|
if (newOff > oldOff)
|
|
{
|
|
for (i=0; i<SUPERLEVEL; i+=realRange)
|
|
myClearGpioOff(gpio, i+oldOff);
|
|
}
|
|
else
|
|
{
|
|
gpioInfo[gpio].deferOff = oldOff;
|
|
gpioInfo[gpio].deferRng = realRange;
|
|
}
|
|
}
|
|
else if (newOff) /* PWM START */
|
|
{
|
|
if (newOff != realRange)
|
|
{
|
|
for (i=0; i<SUPERLEVEL; i+=realRange) mySetGpioOff(gpio, i+newOff);
|
|
}
|
|
|
|
/* schedule new gpio on */
|
|
|
|
for (i=0; i<SUPERCYCLE; i+=cycles) mySetGpioOn(gpio, i);
|
|
}
|
|
else /* PWM STOP */
|
|
{
|
|
/* deschedule gpio on */
|
|
|
|
for (i=0; i<SUPERCYCLE; i+=cycles)
|
|
myClearGpioOn(gpio, i);
|
|
|
|
for (i=0; i<SUPERLEVEL; i+=realRange)
|
|
myClearGpioOff(gpio, i+oldOff);
|
|
|
|
switchGpioOff = 1;
|
|
}
|
|
|
|
if (switchGpioOff)
|
|
{
|
|
*(gpioReg + GPCLR0) = (1<<gpio);
|
|
*(gpioReg + GPCLR0) = (1<<gpio);
|
|
}
|
|
}
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static void myGpioSetServo(unsigned gpio, int oldVal, int newVal)
|
|
{
|
|
int newOff, oldOff, realRange, cycles, i;
|
|
int deferOff, deferRng;
|
|
|
|
DBG(DBG_INTERNAL,
|
|
"myGpioSetServo %d from %d to %d", gpio, oldVal, newVal);
|
|
|
|
realRange = pwmRealRange[clkCfg[gpioCfg.clockMicros].servoIdx];
|
|
cycles = pwmCycles [clkCfg[gpioCfg.clockMicros].servoIdx];
|
|
|
|
newOff = (newVal * realRange)/20000;
|
|
oldOff = (oldVal * realRange)/20000;
|
|
|
|
deferOff = gpioInfo[gpio].deferOff;
|
|
deferRng = gpioInfo[gpio].deferRng;
|
|
|
|
if (gpioInfo[gpio].deferOff)
|
|
{
|
|
for (i=0; i<SUPERLEVEL; i+=deferRng)
|
|
{
|
|
myClearGpioOff(gpio, i+deferOff);
|
|
}
|
|
gpioInfo[gpio].deferOff = 0;
|
|
}
|
|
|
|
if (newOff != oldOff)
|
|
{
|
|
if (newOff && oldOff) /* SERVO CHANGE */
|
|
{
|
|
for (i=0; i<SUPERLEVEL; i+=realRange)
|
|
mySetGpioOff(gpio, i+newOff);
|
|
|
|
if (newOff > oldOff)
|
|
{
|
|
for (i=0; i<SUPERLEVEL; i+=realRange)
|
|
myClearGpioOff(gpio, i+oldOff);
|
|
}
|
|
else
|
|
{
|
|
gpioInfo[gpio].deferOff = oldOff;
|
|
gpioInfo[gpio].deferRng = realRange;
|
|
}
|
|
}
|
|
else if (newOff) /* SERVO START */
|
|
{
|
|
for (i=0; i<SUPERLEVEL; i+=realRange)
|
|
mySetGpioOff(gpio, i+newOff);
|
|
|
|
/* schedule new gpio on */
|
|
|
|
for (i=0; i<SUPERCYCLE; i+=cycles) mySetGpioOn(gpio, i);
|
|
}
|
|
else /* SERVO STOP */
|
|
{
|
|
/* deschedule gpio on */
|
|
|
|
for (i=0; i<SUPERCYCLE; i+=cycles)
|
|
myClearGpioOn(gpio, i);
|
|
|
|
/* if in pulse then delay for the last cycle to complete */
|
|
|
|
if (myGpioRead(gpio)) myGpioDelay(PI_MAX_SERVO_PULSEWIDTH);
|
|
|
|
/* deschedule gpio off */
|
|
|
|
for (i=0; i<SUPERLEVEL; i+=realRange)
|
|
myClearGpioOff(gpio, i+oldOff);
|
|
}
|
|
}
|
|
}
|
|
|
|
/* ======================================================================= */
|
|
|
|
/*
|
|
https://github.com/raspberrypi/firmware/wiki/Mailbox-property-interface
|
|
*/
|
|
|
|
static int mbCreate(char *dev)
|
|
{
|
|
/* <0 error */
|
|
|
|
unlink(dev);
|
|
|
|
return mknod(dev, S_IFCHR|0600, makedev(MB_DEV_MAJOR, 0));
|
|
}
|
|
|
|
static int mbOpen(void)
|
|
{
|
|
/* <0 error */
|
|
|
|
int fd;
|
|
|
|
fd = open(MB_DEV1, 0);
|
|
|
|
if (fd < 0)
|
|
{
|
|
mbCreate(MB_DEV2);
|
|
fd = open(MB_DEV2, 0);
|
|
}
|
|
return fd;
|
|
}
|
|
|
|
static void mbClose(int fd)
|
|
{
|
|
close(fd);
|
|
}
|
|
|
|
static int mbProperty(int fd, void *buf)
|
|
{
|
|
return ioctl(fd, MB_IOCTL, buf);
|
|
}
|
|
|
|
static unsigned mbAllocateMemory(
|
|
int fd, unsigned size, unsigned align, unsigned flags)
|
|
{
|
|
int i=1;
|
|
unsigned p[32];
|
|
|
|
p[i++] = MB_PROCESS_REQUEST;
|
|
p[i++] = MB_ALLOCATE_MEMORY_TAG;
|
|
p[i++] = 12;
|
|
p[i++] = 12;
|
|
p[i++] = size;
|
|
p[i++] = align;
|
|
p[i++] = flags;
|
|
p[i++] = MB_END_TAG;
|
|
p[0] = i*sizeof(*p);
|
|
|
|
mbProperty(fd, p);
|
|
|
|
return p[5];
|
|
}
|
|
|
|
static unsigned mbLockMemory(int fd, unsigned handle)
|
|
{
|
|
int i=1;
|
|
unsigned p[32];
|
|
|
|
p[i++] = MB_PROCESS_REQUEST;
|
|
p[i++] = MB_LOCK_MEMORY_TAG;
|
|
p[i++] = 4;
|
|
p[i++] = 4;
|
|
p[i++] = handle;
|
|
p[i++] = MB_END_TAG;
|
|
p[0] = i*sizeof(*p);
|
|
|
|
mbProperty(fd, p);
|
|
|
|
return p[5];
|
|
}
|
|
|
|
static unsigned mbUnlockMemory(int fd, unsigned handle)
|
|
{
|
|
int i=1;
|
|
unsigned p[32];
|
|
|
|
p[i++] = MB_PROCESS_REQUEST;
|
|
p[i++] = MB_UNLOCK_MEMORY_TAG;
|
|
p[i++] = 4;
|
|
p[i++] = 4;
|
|
p[i++] = handle;
|
|
p[i++] = MB_END_TAG;
|
|
p[0] = i*sizeof(*p);
|
|
|
|
mbProperty(fd, p);
|
|
|
|
return p[5];
|
|
}
|
|
|
|
static unsigned mbReleaseMemory(int fd, unsigned handle)
|
|
{
|
|
int i=1;
|
|
unsigned p[32];
|
|
|
|
p[i++] = MB_PROCESS_REQUEST;
|
|
p[i++] = MB_RELEASE_MEMORY_TAG;
|
|
p[i++] = 4;
|
|
p[i++] = 4;
|
|
p[i++] = handle;
|
|
p[i++] = MB_END_TAG;
|
|
p[0] = i*sizeof(*p);
|
|
|
|
mbProperty(fd, p);
|
|
|
|
return p[5];
|
|
}
|
|
|
|
static void *mbMapMem(unsigned base, unsigned size)
|
|
{
|
|
void *mem = MAP_FAILED;
|
|
|
|
mem = mmap(0, size, PROT_READ|PROT_WRITE, MAP_SHARED, fdMem, base);
|
|
|
|
return mem;
|
|
}
|
|
|
|
static int mbUnmapMem(void *addr, unsigned size)
|
|
{
|
|
/* 0 okay, -1 fail */
|
|
return munmap(addr, size);
|
|
}
|
|
|
|
static void mbDMAFree(DMAMem_t *DMAMemP)
|
|
{
|
|
if (DMAMemP->handle)
|
|
{
|
|
mbUnmapMem(DMAMemP->virtual_addr, DMAMemP->size);
|
|
mbUnlockMemory(fdMbox, DMAMemP->handle);
|
|
mbReleaseMemory(fdMbox, DMAMemP->handle);
|
|
DMAMemP->handle = 0;
|
|
}
|
|
}
|
|
|
|
static int mbDMAAlloc(DMAMem_t *DMAMemP, unsigned size, uint32_t pi_mem_flag)
|
|
{
|
|
DMAMemP->size = size;
|
|
|
|
DMAMemP->handle =
|
|
mbAllocateMemory(fdMbox, size, PAGE_SIZE, pi_mem_flag);
|
|
|
|
if (DMAMemP->handle)
|
|
{
|
|
DMAMemP->bus_addr = mbLockMemory(fdMbox, DMAMemP->handle);
|
|
|
|
DMAMemP->virtual_addr =
|
|
mbMapMem(BUS_TO_PHYS(DMAMemP->bus_addr), size);
|
|
|
|
return 1;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
|
|
/* ======================================================================= */
|
|
|
|
rawCbs_t * rawWaveCBAdr(int cbNum)
|
|
{
|
|
int page, slot;
|
|
|
|
page = cbNum/CBS_PER_OPAGE;
|
|
slot = cbNum%CBS_PER_OPAGE;
|
|
|
|
return &dmaOVirt[page]->cb[slot];
|
|
}
|
|
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static uint32_t waveCbPOadr(int pos)
|
|
{
|
|
int page, slot;
|
|
|
|
page = pos/CBS_PER_OPAGE;
|
|
slot = pos%CBS_PER_OPAGE;
|
|
|
|
//cast twice to suppress compiler warning, I belive this cast is ok
|
|
//because dmaOBus contains bus addresses, not virtual addresses.
|
|
return (uint32_t)(uintptr_t) &dmaOBus[page]->cb[slot];
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static void waveOOLPageSlot(int pos, int *page, int *slot)
|
|
{
|
|
*page = pos/OOL_PER_OPAGE;
|
|
*slot = pos%OOL_PER_OPAGE;
|
|
}
|
|
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static void waveSetOOL(int pos, uint32_t OOL)
|
|
{
|
|
int page, slot;
|
|
|
|
waveOOLPageSlot(pos, &page, &slot);
|
|
|
|
dmaOVirt[page]->OOL[slot] = OOL;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static uint32_t waveOOLPOadr(int pos)
|
|
{
|
|
int page, slot;
|
|
|
|
waveOOLPageSlot(pos, &page, &slot);
|
|
|
|
//cast twice to suppress compiler warning, I belive this cast is ok
|
|
//because dmaOBus contains bus addresses, not virtual addresses.
|
|
return (uint32_t)(uintptr_t) &dmaOBus[page]->OOL[slot];
|
|
}
|
|
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static void waveBitDelay
|
|
(unsigned baud, unsigned bits, unsigned stops, unsigned *bitDelay)
|
|
{
|
|
unsigned fullBit, last, diff, t, i;
|
|
|
|
/* scaled 1000X */
|
|
|
|
fullBit = 1000000000 / baud;
|
|
last = 0;
|
|
|
|
for (i=0; i<=bits; i++)
|
|
{
|
|
t = (((i+1)*fullBit)+500)/1000;
|
|
diff = t - last;
|
|
last = t;
|
|
bitDelay[i] = diff;
|
|
}
|
|
|
|
t = (((bits+1)*fullBit) + ((stops*fullBit)/2) + 500)/1000;
|
|
diff = t - last;
|
|
bitDelay[i] = diff;
|
|
}
|
|
|
|
static int waveDelayCBs(uint32_t delay)
|
|
{
|
|
uint32_t cbs;
|
|
|
|
if (!delay) return 0;
|
|
if (gpioCfg.DMAsecondaryChannel < DMA_LITE_FIRST) return 1;
|
|
cbs = BPD * delay / DMA_LITE_MAX;
|
|
if ((BPD * delay) % DMA_LITE_MAX) cbs++;
|
|
return cbs;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static void waveCBsOOLs(int *numCBs, int *numBOOLs, int *numTOOLs)
|
|
{
|
|
int numCB=0, numBOOL=0, numTOOL=0;
|
|
|
|
unsigned i;
|
|
|
|
unsigned numWaves;
|
|
|
|
rawWave_t *waves;
|
|
|
|
numWaves = wfc[wfcur];
|
|
waves = wf [wfcur];
|
|
|
|
/* delay cb at start of DMA */
|
|
|
|
numCB++;
|
|
|
|
for (i=0; i<numWaves; i++)
|
|
{
|
|
if (waves[i].gpioOn) {numBOOL++;}
|
|
if (waves[i].gpioOff) {numBOOL++;}
|
|
if (waves[i].gpioOn || waves[i].gpioOff) {numCB++;}
|
|
if (waves[i].flags & WAVE_FLAG_READ) {numCB++; numTOOL++;}
|
|
if (waves[i].flags & WAVE_FLAG_TICK) {numCB++; numTOOL++;}
|
|
|
|
numCB += waveDelayCBs(waves[i].usDelay);
|
|
}
|
|
|
|
*numCBs = numCB;
|
|
*numBOOLs = numBOOL;
|
|
*numTOOLs = numTOOL;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static int wave2Cbs(unsigned wave_mode, int *CB, int *BOOL, int *TOOL,
|
|
int numCB, int numBOOL, int numTOOL)
|
|
{
|
|
int botCB=*CB, botOOL=*BOOL, topOOL=*TOOL;
|
|
|
|
int status, s_stride;
|
|
|
|
rawCbs_t *p=NULL;
|
|
|
|
unsigned i, repeatCB;
|
|
|
|
unsigned numWaves;
|
|
|
|
unsigned delayCBs, dcb;
|
|
|
|
uint32_t delayLeft;
|
|
|
|
rawWave_t * waves;
|
|
|
|
numWaves = wfc[wfcur];
|
|
waves = wf [wfcur];
|
|
|
|
/* add delay cb at start of DMA */
|
|
|
|
p = rawWaveCBAdr(botCB++);
|
|
|
|
/* use the secondary clock */
|
|
|
|
if (gpioCfg.clockPeriph != PI_CLOCK_PCM)
|
|
{
|
|
p->info = NORMAL_DMA | TIMED_DMA(2);
|
|
p->dst = PCM_TIMER;
|
|
}
|
|
else
|
|
{
|
|
p->info = NORMAL_DMA | TIMED_DMA(5);
|
|
p->dst = PWM_TIMER;
|
|
}
|
|
|
|
//cast twice to suppress compiler warning, I belive this cast is ok
|
|
//because dmaOBus contains bus addresses, not virtual addresses.
|
|
p->src = (uint32_t)(uintptr_t) (&dmaOBus[0]->periphData);
|
|
p->length = BPD * 20 / PI_WF_MICROS; /* 20 micros delay */
|
|
p->next = waveCbPOadr(botCB);
|
|
|
|
repeatCB = botCB;
|
|
|
|
for (i=0; i<numWaves; i++)
|
|
{
|
|
if (waves[i].gpioOn && waves[i].gpioOff)
|
|
/* Use 2-beat burst */
|
|
{
|
|
p = rawWaveCBAdr(botCB++);
|
|
|
|
p->info = TWO_BEAT_DMA;
|
|
p->src = waveOOLPOadr(botOOL);
|
|
waveSetOOL(botOOL++, waves[i].gpioOn);
|
|
s_stride = waveOOLPOadr(botOOL) - p->src;
|
|
waveSetOOL(botOOL++, waves[i].gpioOff);
|
|
p->dst = ((GPIO_BASE + (GPSET0*4)) & 0x00ffffff) | PI_PERI_BUS;
|
|
p->length = (2<<16) + 4; // 2 transfers of 4 bytes each
|
|
p->stride = (12<<16) + s_stride; // d_stride = (GPCLR0-GPSET0)*4 = 12
|
|
p->next = waveCbPOadr(botCB);
|
|
}
|
|
if (waves[i].gpioOn && !waves[i].gpioOff)
|
|
{
|
|
waveSetOOL(botOOL, waves[i].gpioOn);
|
|
|
|
p = rawWaveCBAdr(botCB++);
|
|
|
|
p->info = NORMAL_DMA;
|
|
p->src = waveOOLPOadr(botOOL++);
|
|
p->dst = ((GPIO_BASE + (GPSET0*4)) & 0x00ffffff) | PI_PERI_BUS;
|
|
p->length = 4;
|
|
p->next = waveCbPOadr(botCB);
|
|
}
|
|
if (waves[i].gpioOff && !waves[i].gpioOn)
|
|
{
|
|
waveSetOOL(botOOL, waves[i].gpioOff);
|
|
|
|
p = rawWaveCBAdr(botCB++);
|
|
|
|
p->info = NORMAL_DMA;
|
|
p->src = waveOOLPOadr(botOOL++);
|
|
p->dst = ((GPIO_BASE + (GPCLR0*4)) & 0x00ffffff) | PI_PERI_BUS;
|
|
p->length = 4;
|
|
p->next = waveCbPOadr(botCB);
|
|
}
|
|
if (waves[i].flags & WAVE_FLAG_READ)
|
|
{
|
|
p = rawWaveCBAdr(botCB++);
|
|
|
|
p->info = NORMAL_DMA;
|
|
p->src = ((GPIO_BASE + (GPLEV0*4)) & 0x00ffffff) | PI_PERI_BUS;
|
|
p->dst = waveOOLPOadr(--topOOL);
|
|
p->length = 4;
|
|
p->next = waveCbPOadr(botCB);
|
|
}
|
|
|
|
if (waves[i].flags & WAVE_FLAG_TICK)
|
|
{
|
|
p = rawWaveCBAdr(botCB++);
|
|
|
|
p->info = NORMAL_DMA;
|
|
p->src = ((SYST_BASE + (SYST_CLO*4)) & 0x00ffffff) | PI_PERI_BUS;
|
|
p->dst = waveOOLPOadr(--topOOL);
|
|
p->length = 4;
|
|
p->next = waveCbPOadr(botCB);
|
|
}
|
|
|
|
if (waves[i].usDelay)
|
|
{
|
|
delayLeft = waves[i].usDelay;
|
|
|
|
delayCBs = waveDelayCBs(delayLeft);
|
|
|
|
for (dcb=0; dcb<delayCBs; dcb++)
|
|
{
|
|
p = rawWaveCBAdr(botCB++);
|
|
|
|
/* use the secondary clock */
|
|
|
|
if (gpioCfg.clockPeriph != PI_CLOCK_PCM)
|
|
{
|
|
p->info = NORMAL_DMA | TIMED_DMA(2);
|
|
p->dst = PCM_TIMER;
|
|
}
|
|
else
|
|
{
|
|
p->info = NORMAL_DMA | TIMED_DMA(5);
|
|
p->dst = PWM_TIMER;
|
|
}
|
|
|
|
//cast twice to suppress compiler warning, I belive this cast is ok
|
|
//because dmaOBus contains bus addresses, not virtual addresses.
|
|
p->src = (uint32_t)(uintptr_t) (&dmaOBus[0]->periphData);
|
|
|
|
p->length = BPD * delayLeft / PI_WF_MICROS;
|
|
|
|
if ((gpioCfg.DMAsecondaryChannel >= DMA_LITE_FIRST) &&
|
|
(p->length > DMA_LITE_MAX))
|
|
{
|
|
p->length = DMA_LITE_MAX;
|
|
}
|
|
|
|
delayLeft -= (p->length / BPD);
|
|
|
|
p->next = waveCbPOadr(botCB);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (numCB)
|
|
{
|
|
/* Pad the wave */
|
|
|
|
botCB = *CB + numCB - 1;
|
|
botOOL = *BOOL + numBOOL - 1;
|
|
topOOL = *TOOL - numTOOL;
|
|
|
|
/* Link the last CB to end of wave */
|
|
|
|
p->next = waveCbPOadr(botCB);
|
|
|
|
/* Insert sentinel CB at end of DMA */
|
|
|
|
p = rawWaveCBAdr(botCB++);
|
|
p->info = NORMAL_DMA | DMA_DEST_IGNORE;
|
|
p->src = waveOOLPOadr(botOOL++);
|
|
p->dst = ((GPIO_BASE + (GPSET0*4)) & 0x00ffffff) | PI_PERI_BUS;
|
|
p->length = 4;
|
|
p->next = 0;
|
|
}
|
|
|
|
if (p != NULL)
|
|
{
|
|
if (wave_mode == PI_WAVE_MODE_ONE_SHOT)
|
|
p->next = 0;
|
|
else p->next = waveCbPOadr(repeatCB);
|
|
}
|
|
|
|
status = botCB - *CB;
|
|
|
|
*CB = botCB;
|
|
*BOOL = botOOL;
|
|
*TOOL = topOOL;
|
|
|
|
return status;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static void waveRxSerial(wfRx_t *w, int level, uint32_t tick)
|
|
{
|
|
int diffTicks, lastLevel;
|
|
int newWritePos;
|
|
|
|
level = level ^ w->s.invert;
|
|
|
|
if (w->s.bit >= 0)
|
|
{
|
|
diffTicks = tick - w->s.startBitTick;
|
|
|
|
if (level != PI_TIMEOUT)
|
|
{
|
|
w->s.level = level;
|
|
lastLevel = !level;
|
|
}
|
|
else lastLevel = w->s.level;
|
|
|
|
while ((w->s.bit <= w->s.dataBits) &&
|
|
(diffTicks > (w->s.nextBitDiff/1000)))
|
|
{
|
|
if (w->s.bit)
|
|
{
|
|
if (lastLevel) w->s.data |= (1<<(w->s.bit-1));
|
|
}
|
|
else w->s.data = 0;
|
|
|
|
++(w->s.bit);
|
|
|
|
w->s.nextBitDiff += w->s.fullBit;
|
|
}
|
|
|
|
if (w->s.bit > w->s.dataBits)
|
|
{
|
|
memcpy(w->s.buf + w->s.writePos, &w->s.data, w->s.bytes);
|
|
|
|
/* don't let writePos catch readPos */
|
|
|
|
newWritePos = (w->s.writePos + w->s.bytes) % (w->s.bufSize);
|
|
|
|
if (newWritePos != w->s.readPos) w->s.writePos = newWritePos;
|
|
|
|
if (level == 0)
|
|
{
|
|
gpioSetWatchdog(w->gpio, w->s.timeout);
|
|
w->s.bit = 0;
|
|
w->s.startBitTick = tick;
|
|
w->s.nextBitDiff = w->s.halfBit;
|
|
}
|
|
else
|
|
{
|
|
w->s.bit = -1;
|
|
gpioSetWatchdog(w->gpio, 0);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
/* start bit if high->low */
|
|
|
|
if (level == 0)
|
|
{
|
|
gpioSetWatchdog(w->gpio, w->s.timeout);
|
|
w->s.level = 0;
|
|
w->s.bit = 0;
|
|
w->s.startBitTick = tick;
|
|
w->s.nextBitDiff = w->s.halfBit;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static void waveRxBit(int gpio, int level, uint32_t tick)
|
|
{
|
|
switch (wfRx[gpio].mode)
|
|
{
|
|
case PI_WFRX_SERIAL:
|
|
waveRxSerial(&wfRx[gpio], level, tick);
|
|
}
|
|
}
|
|
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int rawWaveAddGeneric(unsigned numIn1, rawWave_t *in1)
|
|
{
|
|
unsigned inPos1=0, inPos2=0, outPos=0, level = NUM_WAVE_OOL;
|
|
|
|
unsigned cbs=0;
|
|
|
|
unsigned numIn2, numOut;
|
|
|
|
uint32_t tNow, tNext1, tNext2, tDelay, tMax;
|
|
|
|
rawWave_t *in2, *out;
|
|
|
|
numIn2 = wfc[wfcur];
|
|
in2 = wf[wfcur];
|
|
|
|
numOut = PI_WAVE_MAX_PULSES;
|
|
out = wf[1-wfcur];
|
|
|
|
tNow = 0;
|
|
tMax = 0;
|
|
|
|
if (!numIn1) tNext1 = -1; else tNext1 = 0;
|
|
if (!numIn2) tNext2 = -1; else tNext2 = 0;
|
|
|
|
while (((inPos1<numIn1) || (inPos2<numIn2)) && (outPos<numOut))
|
|
{
|
|
if (tNext1 < tNext2)
|
|
{
|
|
/* pulse 1 due */
|
|
|
|
if (tNow < tNext1)
|
|
{
|
|
/* extend previous delay */
|
|
out[outPos-1].usDelay += (tNext1 - tNow);
|
|
tNow = tNext1;
|
|
}
|
|
|
|
out[outPos].gpioOn = in1[inPos1].gpioOn;
|
|
out[outPos].gpioOff = in1[inPos1].gpioOff;
|
|
out[outPos].flags = in1[inPos1].flags;
|
|
|
|
tNext1 = tNow + in1[inPos1].usDelay; ++inPos1;
|
|
if (tMax < tNext1) tMax = tNext1;
|
|
}
|
|
else if (tNext2 < tNext1)
|
|
{
|
|
/* pulse 2 due */
|
|
|
|
if (tNow < tNext2)
|
|
{
|
|
/* extend previous delay */
|
|
out[outPos-1].usDelay += (tNext2 - tNow);
|
|
tNow = tNext2;
|
|
}
|
|
|
|
out[outPos].gpioOn = in2[inPos2].gpioOn;
|
|
out[outPos].gpioOff = in2[inPos2].gpioOff;
|
|
out[outPos].flags = in2[inPos2].flags;
|
|
|
|
tNext2 = tNow + in2[inPos2].usDelay; ++inPos2;
|
|
if (tMax < tNext2) tMax = tNext2;
|
|
}
|
|
else
|
|
{
|
|
/* pulse 1 and 2 both due */
|
|
|
|
if (tNow < tNext1)
|
|
{
|
|
/* extend previous delay */
|
|
out[outPos-1].usDelay += (tNext1 - tNow);
|
|
tNow = tNext1;
|
|
}
|
|
|
|
out[outPos].gpioOn = in1[inPos1].gpioOn | in2[inPos2].gpioOn;
|
|
out[outPos].gpioOff = in1[inPos1].gpioOff | in2[inPos2].gpioOff;
|
|
out[outPos].flags = in1[inPos1].flags | in2[inPos2].flags;
|
|
|
|
tNext1 = tNow + in1[inPos1].usDelay; ++inPos1;
|
|
tNext2 = tNow + in2[inPos2].usDelay; ++inPos2;
|
|
if (tMax < tNext1) tMax = tNext1;
|
|
if (tMax < tNext2) tMax = tNext2;
|
|
}
|
|
|
|
if (tNext1 <= tNext2) { tDelay = tNext1 - tNow; tNow = tNext1; }
|
|
else { tDelay = tNext2 - tNow; tNow = tNext2; }
|
|
|
|
out[outPos].usDelay = tDelay;
|
|
|
|
cbs += waveDelayCBs(tDelay);
|
|
|
|
if (out[outPos].gpioOn || out[outPos].gpioOff) cbs++;
|
|
|
|
if (out[outPos].flags & WAVE_FLAG_READ)
|
|
{
|
|
cbs++; /* one cb if read */
|
|
--level;
|
|
}
|
|
|
|
if (out[outPos].flags & WAVE_FLAG_TICK)
|
|
{
|
|
cbs++; /* one cb if tick */
|
|
--level;
|
|
}
|
|
|
|
outPos++;
|
|
|
|
if (inPos1 >= numIn1) tNext1 = -1;
|
|
if (inPos2 >= numIn2) tNext2 = -1;
|
|
|
|
}
|
|
|
|
if (tNow < tMax)
|
|
{
|
|
/* extend previous delay */
|
|
out[outPos-1].usDelay += (tMax - tNow);
|
|
tNow = tMax;
|
|
}
|
|
|
|
if ((outPos < numOut) && (outPos < level))
|
|
{
|
|
wfStats.micros = tNow;
|
|
|
|
if (tNow > wfStats.highMicros) wfStats.highMicros = tNow;
|
|
|
|
wfStats.pulses = outPos;
|
|
|
|
if (outPos > wfStats.highPulses) wfStats.highPulses = outPos;
|
|
|
|
wfStats.cbs = cbs;
|
|
|
|
if (cbs > wfStats.highCbs) wfStats.highCbs = cbs;
|
|
|
|
wfc[1-wfcur] = outPos;
|
|
wfcur = 1 - wfcur;
|
|
|
|
return outPos;
|
|
}
|
|
else return PI_TOO_MANY_PULSES;
|
|
}
|
|
|
|
/* ======================================================================= */
|
|
|
|
int i2cWriteQuick(unsigned handle, unsigned bit)
|
|
{
|
|
int status;
|
|
|
|
DBG(DBG_USER, "handle=%d bit=%d", handle, bit);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (handle >= PI_I2C_SLOTS)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
if (i2cInfo[handle].state != PI_I2C_OPENED)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
if ((i2cInfo[handle].funcs & PI_I2C_FUNC_SMBUS_QUICK) == 0)
|
|
SOFT_ERROR(PI_BAD_SMBUS_CMD, "SMBUS command not supported by driver");
|
|
|
|
if (bit > 1)
|
|
SOFT_ERROR(PI_BAD_PARAM, "bad bit (%d)", bit);
|
|
|
|
status = my_smbus_access(
|
|
i2cInfo[handle].fd, bit, 0, PI_I2C_SMBUS_QUICK, NULL);
|
|
|
|
if (status < 0)
|
|
{
|
|
DBG(DBG_USER, "error=%d (%m)", status);
|
|
return PI_I2C_WRITE_FAILED;
|
|
}
|
|
|
|
return status;
|
|
}
|
|
|
|
int i2cReadByte(unsigned handle)
|
|
{
|
|
union my_smbus_data data;
|
|
int status;
|
|
|
|
DBG(DBG_USER, "handle=%d", handle);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (handle >= PI_I2C_SLOTS)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
if (i2cInfo[handle].state != PI_I2C_OPENED)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
if ((i2cInfo[handle].funcs & PI_I2C_FUNC_SMBUS_READ_BYTE) == 0)
|
|
SOFT_ERROR(PI_BAD_SMBUS_CMD, "SMBUS command not supported by driver");
|
|
|
|
status = my_smbus_access(
|
|
i2cInfo[handle].fd, PI_I2C_SMBUS_READ, 0, PI_I2C_SMBUS_BYTE, &data);
|
|
|
|
if (status < 0)
|
|
{
|
|
DBG(DBG_USER, "error=%d (%m)", status);
|
|
return PI_I2C_READ_FAILED;
|
|
}
|
|
|
|
return 0xFF & data.byte;
|
|
}
|
|
|
|
|
|
int i2cWriteByte(unsigned handle, unsigned bVal)
|
|
{
|
|
int status;
|
|
|
|
DBG(DBG_USER, "handle=%d bVal=%d", handle, bVal);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (handle >= PI_I2C_SLOTS)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
if (i2cInfo[handle].state != PI_I2C_OPENED)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
if ((i2cInfo[handle].funcs & PI_I2C_FUNC_SMBUS_WRITE_BYTE) == 0)
|
|
SOFT_ERROR(PI_BAD_SMBUS_CMD, "SMBUS command not supported by driver");
|
|
|
|
if (bVal > 0xFF)
|
|
SOFT_ERROR(PI_BAD_PARAM, "bad bVal (%d)", bVal);
|
|
|
|
status = my_smbus_access(
|
|
i2cInfo[handle].fd,
|
|
PI_I2C_SMBUS_WRITE,
|
|
bVal,
|
|
PI_I2C_SMBUS_BYTE,
|
|
NULL);
|
|
|
|
if (status < 0)
|
|
{
|
|
DBG(DBG_USER, "error=%d (%m)", status);
|
|
return PI_I2C_WRITE_FAILED;
|
|
}
|
|
|
|
return status;
|
|
}
|
|
|
|
|
|
int i2cReadByteData(unsigned handle, unsigned reg)
|
|
{
|
|
union my_smbus_data data;
|
|
int status;
|
|
|
|
DBG(DBG_USER, "handle=%d reg=%d", handle, reg);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (handle >= PI_I2C_SLOTS)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
if (i2cInfo[handle].state != PI_I2C_OPENED)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
if ((i2cInfo[handle].funcs & PI_I2C_FUNC_SMBUS_READ_BYTE_DATA) == 0)
|
|
SOFT_ERROR(PI_BAD_SMBUS_CMD, "SMBUS command not supported by driver");
|
|
|
|
if (reg > 0xFF)
|
|
SOFT_ERROR(PI_BAD_PARAM, "bad reg (%d)", reg);
|
|
|
|
status = my_smbus_access(i2cInfo[handle].fd,
|
|
PI_I2C_SMBUS_READ, reg, PI_I2C_SMBUS_BYTE_DATA, &data);
|
|
|
|
if (status < 0)
|
|
{
|
|
DBG(DBG_USER, "error=%d (%m)", status);
|
|
return PI_I2C_READ_FAILED;
|
|
}
|
|
|
|
return 0xFF & data.byte;
|
|
}
|
|
|
|
|
|
int i2cWriteByteData(unsigned handle, unsigned reg, unsigned bVal)
|
|
{
|
|
union my_smbus_data data;
|
|
|
|
int status;
|
|
|
|
DBG(DBG_USER, "handle=%d reg=%d bVal=%d", handle, reg, bVal);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (handle >= PI_I2C_SLOTS)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
if (i2cInfo[handle].state != PI_I2C_OPENED)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
if ((i2cInfo[handle].funcs & PI_I2C_FUNC_SMBUS_WRITE_BYTE_DATA) == 0)
|
|
SOFT_ERROR(PI_BAD_SMBUS_CMD, "SMBUS command not supported by driver");
|
|
|
|
if (reg > 0xFF)
|
|
SOFT_ERROR(PI_BAD_PARAM, "bad reg (%d)", reg);
|
|
|
|
if (bVal > 0xFF)
|
|
SOFT_ERROR(PI_BAD_PARAM, "bad bVal (%d)", bVal);
|
|
|
|
data.byte = bVal;
|
|
|
|
status = my_smbus_access(
|
|
i2cInfo[handle].fd,
|
|
PI_I2C_SMBUS_WRITE,
|
|
reg,
|
|
PI_I2C_SMBUS_BYTE_DATA,
|
|
&data);
|
|
|
|
if (status < 0)
|
|
{
|
|
DBG(DBG_USER, "error=%d (%m)", status);
|
|
return PI_I2C_WRITE_FAILED;
|
|
}
|
|
|
|
return status;
|
|
}
|
|
|
|
|
|
int i2cReadWordData(unsigned handle, unsigned reg)
|
|
{
|
|
union my_smbus_data data;
|
|
int status;
|
|
|
|
DBG(DBG_USER, "handle=%d reg=%d", handle, reg);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (handle >= PI_I2C_SLOTS)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
if (i2cInfo[handle].state != PI_I2C_OPENED)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
if ((i2cInfo[handle].funcs & PI_I2C_FUNC_SMBUS_READ_WORD_DATA) == 0)
|
|
SOFT_ERROR(PI_BAD_SMBUS_CMD, "SMBUS command not supported by driver");
|
|
|
|
if (reg > 0xFF)
|
|
SOFT_ERROR(PI_BAD_PARAM, "bad reg (%d)", reg);
|
|
|
|
status = (my_smbus_access(
|
|
i2cInfo[handle].fd,
|
|
PI_I2C_SMBUS_READ,
|
|
reg,
|
|
PI_I2C_SMBUS_WORD_DATA,
|
|
&data));
|
|
|
|
if (status < 0)
|
|
{
|
|
DBG(DBG_USER, "error=%d (%m)", status);
|
|
return PI_I2C_READ_FAILED;
|
|
}
|
|
|
|
return 0xFFFF & data.word;
|
|
}
|
|
|
|
|
|
int i2cWriteWordData(unsigned handle, unsigned reg, unsigned wVal)
|
|
{
|
|
union my_smbus_data data;
|
|
|
|
int status;
|
|
|
|
DBG(DBG_USER, "handle=%d reg=%d wVal=%d", handle, reg, wVal);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (handle >= PI_I2C_SLOTS)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
if (i2cInfo[handle].state != PI_I2C_OPENED)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
if ((i2cInfo[handle].funcs & PI_I2C_FUNC_SMBUS_WRITE_WORD_DATA) == 0)
|
|
SOFT_ERROR(PI_BAD_SMBUS_CMD, "SMBUS command not supported by driver");
|
|
|
|
if (reg > 0xFF)
|
|
SOFT_ERROR(PI_BAD_PARAM, "bad reg (%d)", reg);
|
|
|
|
if (wVal > 0xFFFF)
|
|
SOFT_ERROR(PI_BAD_PARAM, "bad wVal (%d)", wVal);
|
|
|
|
data.word = wVal;
|
|
|
|
status = my_smbus_access(
|
|
i2cInfo[handle].fd,
|
|
PI_I2C_SMBUS_WRITE,
|
|
reg,
|
|
PI_I2C_SMBUS_WORD_DATA,
|
|
&data);
|
|
|
|
if (status < 0)
|
|
{
|
|
DBG(DBG_USER, "error=%d (%m)", status);
|
|
return PI_I2C_WRITE_FAILED;
|
|
}
|
|
|
|
return status;
|
|
}
|
|
|
|
|
|
int i2cProcessCall(unsigned handle, unsigned reg, unsigned wVal)
|
|
{
|
|
union my_smbus_data data;
|
|
int status;
|
|
|
|
DBG(DBG_USER, "handle=%d reg=%d wVal=%d", handle, reg, wVal);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (handle >= PI_I2C_SLOTS)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
if (i2cInfo[handle].state != PI_I2C_OPENED)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
if ((i2cInfo[handle].funcs & PI_I2C_FUNC_SMBUS_PROC_CALL) == 0)
|
|
SOFT_ERROR(PI_BAD_SMBUS_CMD, "SMBUS command not supported by driver");
|
|
|
|
if (reg > 0xFF)
|
|
SOFT_ERROR(PI_BAD_PARAM, "bad reg (%d)", reg);
|
|
|
|
if (wVal > 0xFFFF)
|
|
SOFT_ERROR(PI_BAD_PARAM, "bad wVal (%d)", wVal);
|
|
|
|
data.word = wVal;
|
|
|
|
status = (my_smbus_access(
|
|
i2cInfo[handle].fd,
|
|
PI_I2C_SMBUS_WRITE,
|
|
reg, PI_I2C_SMBUS_PROC_CALL,
|
|
&data));
|
|
|
|
if (status < 0)
|
|
{
|
|
DBG(DBG_USER, "error=%d (%m)", status);
|
|
return PI_I2C_READ_FAILED;
|
|
}
|
|
|
|
return 0xFFFF & data.word;
|
|
}
|
|
|
|
|
|
int i2cReadBlockData(unsigned handle, unsigned reg, char *buf)
|
|
{
|
|
union my_smbus_data data;
|
|
|
|
int i, status;
|
|
|
|
DBG(DBG_USER, "handle=%d reg=%d buf=%08"PRIXPTR, handle, reg, (uintptr_t)buf);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (handle >= PI_I2C_SLOTS)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
if (i2cInfo[handle].state != PI_I2C_OPENED)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
if ((i2cInfo[handle].funcs & PI_I2C_FUNC_SMBUS_READ_BLOCK_DATA) == 0)
|
|
SOFT_ERROR(PI_BAD_SMBUS_CMD, "SMBUS command not supported by driver");
|
|
|
|
if (reg > 0xFF)
|
|
SOFT_ERROR(PI_BAD_PARAM, "bad reg (%d)", reg);
|
|
|
|
status = (my_smbus_access(
|
|
i2cInfo[handle].fd,
|
|
PI_I2C_SMBUS_READ,
|
|
reg,
|
|
PI_I2C_SMBUS_BLOCK_DATA,
|
|
&data));
|
|
|
|
if (status < 0)
|
|
{
|
|
DBG(DBG_USER, "error=%d (%m)", status);
|
|
return PI_I2C_READ_FAILED;
|
|
}
|
|
else
|
|
{
|
|
if (data.block[0] <= PI_I2C_SMBUS_BLOCK_MAX)
|
|
{
|
|
for (i=0; i<data.block[0]; i++) buf[i] = data.block[i+1];
|
|
return data.block[0];
|
|
}
|
|
else return PI_I2C_READ_FAILED;
|
|
}
|
|
}
|
|
|
|
|
|
int i2cWriteBlockData(
|
|
unsigned handle, unsigned reg, char *buf, unsigned count)
|
|
{
|
|
union my_smbus_data data;
|
|
|
|
int i, status;
|
|
|
|
DBG(DBG_USER, "handle=%d reg=%d count=%d [%s]",
|
|
handle, reg, count, myBuf2Str(count, buf));
|
|
|
|
CHECK_INITED;
|
|
|
|
if (handle >= PI_I2C_SLOTS)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
if (i2cInfo[handle].state != PI_I2C_OPENED)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
if ((i2cInfo[handle].funcs & PI_I2C_FUNC_SMBUS_WRITE_BLOCK_DATA) == 0)
|
|
SOFT_ERROR(PI_BAD_SMBUS_CMD, "SMBUS command not supported by driver");
|
|
|
|
if (reg > 0xFF)
|
|
SOFT_ERROR(PI_BAD_PARAM, "bad reg (%d)", reg);
|
|
|
|
if ((count < 1) || (count > 32))
|
|
SOFT_ERROR(PI_BAD_PARAM, "bad count (%d)", count);
|
|
|
|
for (i=1; i<=count; i++) data.block[i] = buf[i-1];
|
|
data.block[0] = count;
|
|
|
|
status = my_smbus_access(
|
|
i2cInfo[handle].fd,
|
|
PI_I2C_SMBUS_WRITE,
|
|
reg,
|
|
PI_I2C_SMBUS_BLOCK_DATA,
|
|
&data);
|
|
|
|
if (status < 0)
|
|
{
|
|
DBG(DBG_USER, "error=%d (%m)", status);
|
|
return PI_I2C_WRITE_FAILED;
|
|
}
|
|
|
|
return status;
|
|
}
|
|
|
|
|
|
int i2cBlockProcessCall(
|
|
unsigned handle, unsigned reg, char *buf, unsigned count)
|
|
{
|
|
union my_smbus_data data;
|
|
|
|
int i, status;
|
|
|
|
DBG(DBG_USER, "handle=%d reg=%d count=%d [%s]",
|
|
handle, reg, count, myBuf2Str(count, buf));
|
|
|
|
CHECK_INITED;
|
|
|
|
if (handle >= PI_I2C_SLOTS)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
if (i2cInfo[handle].state != PI_I2C_OPENED)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
if ((i2cInfo[handle].funcs & PI_I2C_FUNC_SMBUS_PROC_CALL) == 0)
|
|
SOFT_ERROR(PI_BAD_SMBUS_CMD, "SMBUS command not supported by driver");
|
|
|
|
if (reg > 0xFF)
|
|
SOFT_ERROR(PI_BAD_PARAM, "bad reg (%d)", reg);
|
|
|
|
if ((count < 1) || (count > 32))
|
|
SOFT_ERROR(PI_BAD_PARAM, "bad count (%d)", count);
|
|
|
|
for (i=1; i<=count; i++) data.block[i] = buf[i-1];
|
|
data.block[0] = count;
|
|
|
|
status = (my_smbus_access(
|
|
i2cInfo[handle].fd, PI_I2C_SMBUS_WRITE, reg,
|
|
PI_I2C_SMBUS_BLOCK_PROC_CALL, &data));
|
|
|
|
if (status < 0)
|
|
{
|
|
DBG(DBG_USER, "error=%d (%m)", status);
|
|
return PI_I2C_READ_FAILED;
|
|
}
|
|
else
|
|
{
|
|
if (data.block[0] <= PI_I2C_SMBUS_BLOCK_MAX)
|
|
{
|
|
for (i=0; i<data.block[0]; i++) buf[i] = data.block[i+1];
|
|
return data.block[0];
|
|
}
|
|
else return PI_I2C_READ_FAILED;
|
|
}
|
|
}
|
|
|
|
|
|
int i2cReadI2CBlockData(
|
|
unsigned handle, unsigned reg, char *buf, unsigned count)
|
|
{
|
|
union my_smbus_data data;
|
|
|
|
int i, status;
|
|
uint32_t size;
|
|
|
|
DBG(DBG_USER, "handle=%d reg=%d count=%d buf=%08"PRIXPTR,
|
|
handle, reg, count, (uintptr_t)buf);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (handle >= PI_I2C_SLOTS)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
if (i2cInfo[handle].state != PI_I2C_OPENED)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
if ((i2cInfo[handle].funcs & PI_I2C_FUNC_SMBUS_READ_I2C_BLOCK) == 0)
|
|
SOFT_ERROR(PI_BAD_SMBUS_CMD, "SMBUS command not supported by driver");
|
|
|
|
if (reg > 0xFF)
|
|
SOFT_ERROR(PI_BAD_PARAM, "bad reg (%d)", reg);
|
|
|
|
if ((count < 1) || (count > 32))
|
|
SOFT_ERROR(PI_BAD_PARAM, "bad count (%d)", count);
|
|
|
|
if (count == 32)
|
|
size = PI_I2C_SMBUS_I2C_BLOCK_BROKEN;
|
|
else
|
|
size = PI_I2C_SMBUS_I2C_BLOCK_DATA;
|
|
|
|
data.block[0] = count;
|
|
|
|
status = (my_smbus_access(
|
|
i2cInfo[handle].fd, PI_I2C_SMBUS_READ, reg, size, &data));
|
|
|
|
if (status < 0)
|
|
{
|
|
DBG(DBG_USER, "error=%d (%m)", status);
|
|
return PI_I2C_READ_FAILED;
|
|
}
|
|
else
|
|
{
|
|
if (data.block[0] <= PI_I2C_SMBUS_I2C_BLOCK_MAX)
|
|
{
|
|
for (i=0; i<data.block[0]; i++) buf[i] = data.block[i+1];
|
|
return data.block[0];
|
|
}
|
|
else return PI_I2C_READ_FAILED;
|
|
}
|
|
}
|
|
|
|
|
|
int i2cWriteI2CBlockData(
|
|
unsigned handle, unsigned reg, char *buf, unsigned count)
|
|
{
|
|
union my_smbus_data data;
|
|
|
|
int i, status;
|
|
|
|
DBG(DBG_USER, "handle=%d reg=%d count=%d [%s]",
|
|
handle, reg, count, myBuf2Str(count, buf));
|
|
|
|
CHECK_INITED;
|
|
|
|
if (handle >= PI_I2C_SLOTS)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
if (i2cInfo[handle].state != PI_I2C_OPENED)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
if ((i2cInfo[handle].funcs & PI_I2C_FUNC_SMBUS_WRITE_I2C_BLOCK) == 0)
|
|
SOFT_ERROR(PI_BAD_SMBUS_CMD, "SMBUS command not supported by driver");
|
|
|
|
if (reg > 0xFF)
|
|
SOFT_ERROR(PI_BAD_PARAM, "bad reg (%d)", reg);
|
|
|
|
if ((count < 1) || (count > 32))
|
|
SOFT_ERROR(PI_BAD_PARAM, "bad count (%d)", count);
|
|
|
|
for (i=1; i<=count; i++) data.block[i] = buf[i-1];
|
|
|
|
data.block[0] = count;
|
|
|
|
status = my_smbus_access(
|
|
i2cInfo[handle].fd,
|
|
PI_I2C_SMBUS_WRITE,
|
|
reg,
|
|
PI_I2C_SMBUS_I2C_BLOCK_BROKEN,
|
|
&data);
|
|
|
|
if (status < 0)
|
|
{
|
|
DBG(DBG_USER, "error=%d (%m)", status);
|
|
return PI_I2C_WRITE_FAILED;
|
|
}
|
|
|
|
return status;
|
|
}
|
|
|
|
int i2cWriteDevice(unsigned handle, char *buf, unsigned count)
|
|
{
|
|
int bytes;
|
|
|
|
DBG(DBG_USER, "handle=%d count=%d [%s]",
|
|
handle, count, myBuf2Str(count, buf));
|
|
|
|
CHECK_INITED;
|
|
|
|
if (handle >= PI_I2C_SLOTS)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
if (i2cInfo[handle].state != PI_I2C_OPENED)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
if ((count < 1) || (count > PI_MAX_I2C_DEVICE_COUNT))
|
|
SOFT_ERROR(PI_BAD_PARAM, "bad count (%d)", count);
|
|
|
|
bytes = write(i2cInfo[handle].fd, buf, count);
|
|
|
|
if (bytes != count)
|
|
{
|
|
DBG(DBG_USER, "error=%d (%m)", bytes);
|
|
return PI_I2C_WRITE_FAILED;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int i2cReadDevice(unsigned handle, char *buf, unsigned count)
|
|
{
|
|
int bytes;
|
|
|
|
DBG(DBG_USER, "handle=%d count=%d buf=%08"PRIXPTR,
|
|
handle, count, (uintptr_t)buf);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (handle >= PI_I2C_SLOTS)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
if (i2cInfo[handle].state != PI_I2C_OPENED)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
if ((count < 1) || (count > PI_MAX_I2C_DEVICE_COUNT))
|
|
SOFT_ERROR(PI_BAD_PARAM, "bad count (%d)", count);
|
|
|
|
bytes = read(i2cInfo[handle].fd, buf, count);
|
|
|
|
if (bytes != count)
|
|
{
|
|
DBG(DBG_USER, "error=%d (%m)", bytes);
|
|
return PI_I2C_READ_FAILED;
|
|
}
|
|
|
|
return bytes;
|
|
}
|
|
|
|
int i2cOpen(unsigned i2cBus, unsigned i2cAddr, unsigned i2cFlags)
|
|
{
|
|
static pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER;
|
|
char dev[32];
|
|
int i, slot, fd;
|
|
uint32_t funcs;
|
|
|
|
DBG(DBG_USER, "i2cBus=%d i2cAddr=%d flags=0x%X",
|
|
i2cBus, i2cAddr, i2cFlags);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (i2cAddr > PI_MAX_I2C_ADDR)
|
|
SOFT_ERROR(PI_BAD_I2C_ADDR, "bad I2C address (%d)", i2cAddr);
|
|
|
|
if (i2cFlags)
|
|
SOFT_ERROR(PI_BAD_FLAGS, "bad flags (0x%X)", i2cFlags);
|
|
|
|
slot = -1;
|
|
|
|
pthread_mutex_lock(&mutex);
|
|
|
|
for (i=0; i<PI_I2C_SLOTS; i++)
|
|
{
|
|
if (i2cInfo[i].state == PI_I2C_CLOSED)
|
|
{
|
|
slot = i;
|
|
i2cInfo[slot].state = PI_I2C_RESERVED;
|
|
break;
|
|
}
|
|
}
|
|
|
|
pthread_mutex_unlock(&mutex);
|
|
|
|
if (slot < 0) SOFT_ERROR(PI_NO_HANDLE, "no I2C handles");
|
|
|
|
sprintf(dev, "/dev/i2c-%d", i2cBus);
|
|
|
|
if ((fd = open(dev, O_RDWR)) < 0)
|
|
{
|
|
/* try a modprobe */
|
|
|
|
if (system("/sbin/modprobe i2c_dev") == -1) { /* ignore errors */}
|
|
if (system("/sbin/modprobe i2c_bcm2835") == -1) { /* ignore errors */}
|
|
|
|
myGpioDelay(100000);
|
|
|
|
if ((fd = open(dev, O_RDWR)) < 0)
|
|
{
|
|
i2cInfo[slot].state = PI_I2C_CLOSED;
|
|
return PI_BAD_I2C_BUS;
|
|
}
|
|
}
|
|
|
|
if (ioctl(fd, PI_I2C_SLAVE, i2cAddr) < 0)
|
|
{
|
|
close(fd);
|
|
i2cInfo[slot].state = PI_I2C_CLOSED;
|
|
return PI_I2C_OPEN_FAILED;
|
|
}
|
|
|
|
if (ioctl(fd, PI_I2C_FUNCS, &funcs) < 0)
|
|
{
|
|
funcs = -1; /* assume all smbus commands allowed */
|
|
}
|
|
|
|
i2cInfo[slot].fd = fd;
|
|
i2cInfo[slot].addr = i2cAddr;
|
|
i2cInfo[slot].flags = i2cFlags;
|
|
i2cInfo[slot].funcs = funcs;
|
|
i2cInfo[slot].state = PI_I2C_OPENED;
|
|
|
|
return slot;
|
|
}
|
|
|
|
int i2cClose(unsigned handle)
|
|
{
|
|
DBG(DBG_USER, "handle=%d", handle);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (handle >= PI_I2C_SLOTS)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
if (i2cInfo[handle].state != PI_I2C_OPENED)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
if (i2cInfo[handle].fd >= 0) close(i2cInfo[handle].fd);
|
|
|
|
i2cInfo[handle].fd = -1;
|
|
i2cInfo[handle].state = PI_I2C_CLOSED;
|
|
|
|
return 0;
|
|
}
|
|
|
|
void i2cSwitchCombined(int setting)
|
|
{
|
|
int fd;
|
|
|
|
DBG(DBG_USER, "setting=%d", setting);
|
|
|
|
fd = open(PI_I2C_COMBINED, O_WRONLY);
|
|
|
|
if (fd >= 0)
|
|
{
|
|
if (setting)
|
|
{
|
|
if (write(fd, "1\n", 2) == -1) { /* ignore errors */ }
|
|
}
|
|
else
|
|
{
|
|
if (write(fd, "0\n", 2) == -1) { /* ignore errors */ }
|
|
}
|
|
|
|
close(fd);
|
|
}
|
|
}
|
|
|
|
int i2cSegments(unsigned handle, pi_i2c_msg_t *segs, unsigned numSegs)
|
|
{
|
|
int retval;
|
|
my_i2c_rdwr_ioctl_data_t rdwr;
|
|
|
|
DBG(DBG_USER, "handle=%d", handle);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (handle >= PI_I2C_SLOTS)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
if (i2cInfo[handle].state != PI_I2C_OPENED)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
if (segs == NULL)
|
|
SOFT_ERROR(PI_BAD_POINTER, "null segments");
|
|
|
|
if (numSegs > PI_I2C_RDRW_IOCTL_MAX_MSGS)
|
|
SOFT_ERROR(PI_TOO_MANY_SEGS, "too many segments (%d)", numSegs);
|
|
|
|
rdwr.msgs = segs;
|
|
rdwr.nmsgs = numSegs;
|
|
|
|
retval = ioctl(i2cInfo[handle].fd, PI_I2C_RDWR, &rdwr);
|
|
|
|
if (retval >= 0) return retval;
|
|
else return PI_BAD_I2C_SEG;
|
|
}
|
|
|
|
int i2cZip(
|
|
unsigned handle,
|
|
char *inBuf, unsigned inLen, char *outBuf, unsigned outLen)
|
|
{
|
|
int numSegs, inPos, outPos, status, bytes, flags, addr;
|
|
int esc, setesc;
|
|
pi_i2c_msg_t segs[PI_I2C_RDRW_IOCTL_MAX_MSGS];
|
|
|
|
DBG(DBG_USER, "handle=%d inBuf=%s outBuf=%08"PRIXPTR" len=%d",
|
|
handle, myBuf2Str(inLen, (char *)inBuf), (uintptr_t)outBuf, outLen);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (handle >= PI_I2C_SLOTS)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
if (i2cInfo[handle].state != PI_I2C_OPENED)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
if (!inBuf || !inLen)
|
|
SOFT_ERROR(PI_BAD_POINTER, "input buffer can't be NULL");
|
|
|
|
if (!outBuf && outLen)
|
|
SOFT_ERROR(PI_BAD_POINTER, "output buffer can't be NULL");
|
|
|
|
numSegs = 0;
|
|
|
|
inPos = 0;
|
|
outPos = 0;
|
|
status = 0;
|
|
|
|
addr = i2cInfo[handle].addr;
|
|
flags = 0;
|
|
esc = 0;
|
|
setesc = 0;
|
|
|
|
while (!status && (inPos < inLen))
|
|
{
|
|
DBG(DBG_INTERNAL, "status=%d inpos=%d inlen=%d cmd=%d addr=%d flags=%x",
|
|
status, inPos, inLen, inBuf[inPos], addr, flags);
|
|
|
|
switch (inBuf[inPos++])
|
|
{
|
|
case PI_I2C_END:
|
|
status = 1;
|
|
break;
|
|
|
|
case PI_I2C_COMBINED_ON:
|
|
/* Run prior transactions before setting combined flag */
|
|
if (numSegs)
|
|
{
|
|
status = i2cSegments(handle, segs, numSegs);
|
|
if (status >= 0) status = 0; /* continue */
|
|
numSegs = 0;
|
|
}
|
|
i2cSwitchCombined(1);
|
|
break;
|
|
|
|
case PI_I2C_COMBINED_OFF:
|
|
/* Run prior transactions before clearing combined flag */
|
|
if (numSegs)
|
|
{
|
|
status = i2cSegments(handle, segs, numSegs);
|
|
if (status >= 0) status = 0; /* continue */
|
|
numSegs = 0;
|
|
}
|
|
i2cSwitchCombined(0);
|
|
break;
|
|
|
|
case PI_I2C_ADDR:
|
|
addr = myI2CGetPar(inBuf, &inPos, inLen, &esc);
|
|
if (addr < 0) status = PI_BAD_I2C_CMD;
|
|
break;
|
|
|
|
case PI_I2C_FLAGS:
|
|
/* cheat to force two byte flags */
|
|
esc = 1;
|
|
flags = myI2CGetPar(inBuf, &inPos, inLen, &esc);
|
|
if (flags < 0) status = PI_BAD_I2C_CMD;
|
|
break;
|
|
|
|
case PI_I2C_ESC:
|
|
setesc = 1;
|
|
break;
|
|
|
|
case PI_I2C_READ:
|
|
|
|
bytes = myI2CGetPar(inBuf, &inPos, inLen, &esc);
|
|
|
|
if (bytes >= 0)
|
|
{
|
|
if ((bytes + outPos) < outLen)
|
|
{
|
|
segs[numSegs].addr = addr;
|
|
segs[numSegs].flags = (flags|1);
|
|
segs[numSegs].len = bytes;
|
|
segs[numSegs].buf = (uint8_t *)(outBuf + outPos);
|
|
outPos += bytes;
|
|
numSegs++;
|
|
if (numSegs >= PI_I2C_RDRW_IOCTL_MAX_MSGS)
|
|
{
|
|
status = i2cSegments(handle, segs, numSegs);
|
|
if (status >= 0) status = 0; /* continue */
|
|
numSegs = 0;
|
|
}
|
|
}
|
|
else status = PI_BAD_I2C_RLEN;
|
|
}
|
|
else status = PI_BAD_I2C_RLEN;
|
|
break;
|
|
|
|
case PI_I2C_WRITE:
|
|
|
|
bytes = myI2CGetPar(inBuf, &inPos, inLen, &esc);
|
|
|
|
if (bytes >= 0)
|
|
{
|
|
if ((bytes + inPos) < inLen)
|
|
{
|
|
segs[numSegs].addr = addr;
|
|
segs[numSegs].flags = (flags&0xfffe);
|
|
segs[numSegs].len = bytes;
|
|
segs[numSegs].buf = (uint8_t *)(inBuf + inPos);
|
|
inPos += bytes;
|
|
numSegs++;
|
|
if (numSegs >= PI_I2C_RDRW_IOCTL_MAX_MSGS)
|
|
{
|
|
status = i2cSegments(handle, segs, numSegs);
|
|
if (status >= 0) status = 0; /* continue */
|
|
numSegs = 0;
|
|
}
|
|
}
|
|
else status = PI_BAD_I2C_WLEN;
|
|
}
|
|
else status = PI_BAD_I2C_WLEN;
|
|
break;
|
|
|
|
default:
|
|
status = PI_BAD_I2C_CMD;
|
|
}
|
|
|
|
if (setesc) esc = 1; else esc = 0;
|
|
|
|
setesc = 0;
|
|
}
|
|
|
|
if (status >= 0)
|
|
{
|
|
if (numSegs) status = i2cSegments(handle, segs, numSegs);
|
|
}
|
|
|
|
if (status >= 0) status = outPos;
|
|
|
|
return status;
|
|
}
|
|
|
|
/* ======================================================================= */
|
|
|
|
/*SPI */
|
|
|
|
static uint32_t _spiTXBits(char *buf, int pos, int bitlen, int msbf)
|
|
{
|
|
uint32_t bits=0;
|
|
|
|
if (buf)
|
|
{
|
|
if (bitlen <= 8) bits = *((( uint8_t*)buf)+pos);
|
|
else if (bitlen <= 16) bits = *(((uint16_t*)buf)+pos);
|
|
else bits = *(((uint32_t*)buf)+pos);
|
|
|
|
if (msbf) bits <<= (32-bitlen);
|
|
}
|
|
|
|
return bits;
|
|
}
|
|
|
|
static void _spiRXBits(
|
|
char *buf, int pos, int bitlen, int msbf, uint32_t bits)
|
|
{
|
|
if (buf)
|
|
{
|
|
if (!msbf) bits >>= (32-bitlen);
|
|
|
|
if (bitlen <= 8) *((( uint8_t*)buf)+pos) = bits;
|
|
else if (bitlen <= 16) *(((uint16_t*)buf)+pos) = bits;
|
|
else *(((uint32_t*)buf)+pos) = bits;
|
|
}
|
|
}
|
|
|
|
static void spiACS(int channel, int on)
|
|
{
|
|
int gpio;
|
|
|
|
switch (channel)
|
|
{
|
|
case 0: gpio = PI_ASPI_CE0; break;
|
|
case 1: gpio = PI_ASPI_CE1; break;
|
|
default: gpio = PI_ASPI_CE2; break;
|
|
}
|
|
myGpioWrite(gpio, on);
|
|
}
|
|
|
|
static void spiGoA(
|
|
unsigned speed, /* bits per second */
|
|
uint32_t flags, /* flags */
|
|
char *txBuf, /* tx buffer */
|
|
char *rxBuf, /* rx buffer */
|
|
unsigned count) /* number of bytes */
|
|
{
|
|
int cs;
|
|
char bit_ir[4] = {1, 0, 0, 1}; /* read on rising edge */
|
|
char bit_or[4] = {0, 1, 1, 0}; /* write on rising edge */
|
|
char bit_ic[4] = {0, 0, 1, 1}; /* invert clock */
|
|
|
|
int mode, bitlen, txmsbf, rxmsbf, channel;
|
|
unsigned txCnt=0;
|
|
unsigned rxCnt=0;
|
|
uint32_t spiDefaults;
|
|
uint32_t statusReg;
|
|
int txFull, rxEmpty;
|
|
|
|
channel = PI_SPI_FLAGS_GET_CHANNEL(flags);
|
|
mode = PI_SPI_FLAGS_GET_MODE (flags);
|
|
|
|
bitlen = PI_SPI_FLAGS_GET_BITLEN (flags);
|
|
|
|
if (!bitlen) bitlen = 8;
|
|
|
|
/* correct count for word size */
|
|
|
|
if (bitlen > 8) count /= 2;
|
|
if (bitlen > 16) count /= 2;
|
|
|
|
txmsbf = !PI_SPI_FLAGS_GET_TX_LSB (flags);
|
|
rxmsbf = !PI_SPI_FLAGS_GET_RX_LSB (flags);
|
|
|
|
cs = PI_SPI_FLAGS_GET_CSPOLS(flags) & (1<<channel);
|
|
|
|
spiDefaults = AUXSPI_CNTL0_SPEED((125000000/speed)-1)|
|
|
AUXSPI_CNTL0_IN_RISING(bit_ir[mode]) |
|
|
AUXSPI_CNTL0_OUT_RISING(bit_or[mode]) |
|
|
AUXSPI_CNTL0_INVERT_CLK(bit_ic[mode]) |
|
|
AUXSPI_CNTL0_MSB_FIRST(txmsbf) |
|
|
AUXSPI_CNTL0_SHIFT_LEN(bitlen);
|
|
|
|
if (!count)
|
|
{
|
|
auxReg[AUX_SPI0_CNTL0_REG] =
|
|
AUXSPI_CNTL0_ENABLE | AUXSPI_CNTL0_CLR_FIFOS;
|
|
|
|
myGpioDelay(10);
|
|
|
|
auxReg[AUX_SPI0_CNTL0_REG] = AUXSPI_CNTL0_ENABLE | spiDefaults;
|
|
|
|
auxReg[AUX_SPI0_CNTL1_REG] = AUXSPI_CNTL1_MSB_FIRST(rxmsbf);
|
|
|
|
return;
|
|
}
|
|
|
|
auxReg[AUX_SPI0_CNTL0_REG] = AUXSPI_CNTL0_ENABLE | spiDefaults;
|
|
|
|
auxReg[AUX_SPI0_CNTL1_REG] = AUXSPI_CNTL1_MSB_FIRST(rxmsbf);
|
|
|
|
spiACS(channel, cs);
|
|
|
|
while ((txCnt < count) || (rxCnt < count))
|
|
{
|
|
statusReg = auxReg[AUX_SPI0_STAT_REG];
|
|
|
|
rxEmpty = statusReg & AUXSPI_STAT_RX_EMPTY;
|
|
|
|
txFull = (((statusReg>>28)&15) > 2);
|
|
|
|
if (rxCnt < count)
|
|
{
|
|
if (!rxEmpty)
|
|
{
|
|
_spiRXBits(
|
|
rxBuf, rxCnt++, bitlen, rxmsbf, auxReg[AUX_SPI0_IO_REG]);
|
|
}
|
|
}
|
|
|
|
if (txCnt < count)
|
|
{
|
|
if (!txFull)
|
|
{
|
|
if (txCnt != (count-1))
|
|
{
|
|
auxReg[AUX_SPI0_TX_HOLD] =
|
|
_spiTXBits(txBuf, txCnt++, bitlen, txmsbf);
|
|
}
|
|
else
|
|
{
|
|
auxReg[AUX_SPI0_IO_REG] =
|
|
_spiTXBits(txBuf, txCnt++, bitlen, txmsbf);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
while ((auxReg[AUX_SPI0_STAT_REG] & AUXSPI_STAT_BUSY)) ;
|
|
|
|
spiACS(channel, !cs);
|
|
}
|
|
|
|
static void spiGoS(
|
|
unsigned speed,
|
|
uint32_t flags,
|
|
char *txBuf,
|
|
char *rxBuf,
|
|
unsigned count)
|
|
{
|
|
unsigned txCnt=0;
|
|
unsigned rxCnt=0;
|
|
unsigned cnt, cnt4w, cnt3w;
|
|
uint32_t spiDefaults;
|
|
unsigned mode, channel, cspol, cspols, flag3w, ren3w;
|
|
|
|
channel = PI_SPI_FLAGS_GET_CHANNEL(flags);
|
|
mode = PI_SPI_FLAGS_GET_MODE (flags);
|
|
cspols = PI_SPI_FLAGS_GET_CSPOLS(flags);
|
|
cspol = (cspols>>channel) & 1;
|
|
flag3w = PI_SPI_FLAGS_GET_3WIRE(flags);
|
|
ren3w = PI_SPI_FLAGS_GET_3WREN(flags);
|
|
|
|
spiDefaults = SPI_CS_MODE(mode) |
|
|
SPI_CS_CSPOLS(cspols) |
|
|
SPI_CS_CS(channel) |
|
|
SPI_CS_CSPOL(cspol) |
|
|
SPI_CS_CLEAR(3);
|
|
|
|
spiReg[SPI_DLEN] = 2; /* undocumented, stops inter-byte gap */
|
|
|
|
spiReg[SPI_CS] = spiDefaults; /* stop */
|
|
|
|
if (!count) return;
|
|
|
|
if (flag3w)
|
|
{
|
|
if (ren3w < count)
|
|
{
|
|
cnt4w = ren3w;
|
|
cnt3w = count - ren3w;
|
|
}
|
|
else
|
|
{
|
|
cnt4w = count;
|
|
cnt3w = 0;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
cnt4w = count;
|
|
cnt3w = 0;
|
|
}
|
|
|
|
spiReg[SPI_CLK] = 250000000/speed;
|
|
|
|
spiReg[SPI_CS] = spiDefaults | SPI_CS_TA; /* start */
|
|
|
|
cnt = cnt4w;
|
|
|
|
while((txCnt < cnt) || (rxCnt < cnt))
|
|
{
|
|
while((rxCnt < cnt) && ((spiReg[SPI_CS] & SPI_CS_RXD)))
|
|
{
|
|
if (rxBuf) rxBuf[rxCnt] = spiReg[SPI_FIFO];
|
|
else spi_dummy = spiReg[SPI_FIFO];
|
|
rxCnt++;
|
|
}
|
|
|
|
while((txCnt < cnt) && ((spiReg[SPI_CS] & SPI_CS_TXD)))
|
|
{
|
|
if (txBuf) spiReg[SPI_FIFO] = txBuf[txCnt];
|
|
else spiReg[SPI_FIFO] = 0;
|
|
txCnt++;
|
|
}
|
|
}
|
|
|
|
while (!(spiReg[SPI_CS] & SPI_CS_DONE)) ;
|
|
|
|
/* now switch to 3-wire bus */
|
|
|
|
cnt += cnt3w;
|
|
|
|
spiReg[SPI_CS] |= SPI_CS_REN;
|
|
|
|
while((txCnt < cnt) || (rxCnt < cnt))
|
|
{
|
|
while((rxCnt < cnt) && ((spiReg[SPI_CS] & SPI_CS_RXD)))
|
|
{
|
|
if (rxBuf) rxBuf[rxCnt] = spiReg[SPI_FIFO];
|
|
else spi_dummy = spiReg[SPI_FIFO];
|
|
rxCnt++;
|
|
}
|
|
|
|
while((txCnt < cnt) && ((spiReg[SPI_CS] & SPI_CS_TXD)))
|
|
{
|
|
if (txBuf) spiReg[SPI_FIFO] = txBuf[txCnt];
|
|
else spiReg[SPI_FIFO] = 0;
|
|
txCnt++;
|
|
}
|
|
}
|
|
|
|
while (!(spiReg[SPI_CS] & SPI_CS_DONE)) ;
|
|
|
|
spiReg[SPI_CS] = spiDefaults; /* stop */
|
|
}
|
|
|
|
static void spiGo(
|
|
unsigned speed,
|
|
uint32_t flags,
|
|
char *txBuf,
|
|
char *rxBuf,
|
|
unsigned count)
|
|
{
|
|
static pthread_mutex_t main_mutex = PTHREAD_MUTEX_INITIALIZER;
|
|
static pthread_mutex_t aux_mutex = PTHREAD_MUTEX_INITIALIZER;
|
|
|
|
if (PI_SPI_FLAGS_GET_AUX_SPI(flags))
|
|
{
|
|
pthread_mutex_lock(&aux_mutex);
|
|
spiGoA(speed, flags, txBuf, rxBuf, count);
|
|
pthread_mutex_unlock(&aux_mutex);
|
|
}
|
|
else
|
|
{
|
|
pthread_mutex_lock(&main_mutex);
|
|
spiGoS(speed, flags, txBuf, rxBuf, count);
|
|
pthread_mutex_unlock(&main_mutex);
|
|
}
|
|
}
|
|
|
|
static int spiAnyOpen(uint32_t flags)
|
|
{
|
|
int i, aux;
|
|
|
|
aux = PI_SPI_FLAGS_GET_AUX_SPI(flags);
|
|
|
|
for (i=0; i<PI_SPI_SLOTS; i++)
|
|
{
|
|
if ((spiInfo[i].state == PI_SPI_OPENED) &&
|
|
(PI_SPI_FLAGS_GET_AUX_SPI(spiInfo[i].flags) == aux))
|
|
return 1;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static void spiInit(uint32_t flags)
|
|
{
|
|
uint32_t resvd, cspols;
|
|
|
|
resvd = PI_SPI_FLAGS_GET_RESVD(flags);
|
|
cspols = PI_SPI_FLAGS_GET_CSPOLS(flags);
|
|
|
|
if (PI_SPI_FLAGS_GET_AUX_SPI(flags))
|
|
{
|
|
/* enable module and access to registers */
|
|
|
|
auxReg[AUX_ENABLES] |= AUXENB_SPI1;
|
|
|
|
/* save original state */
|
|
|
|
old_mode_ace0 = gpioGetMode(PI_ASPI_CE0);
|
|
old_mode_ace1 = gpioGetMode(PI_ASPI_CE1);
|
|
old_mode_ace2 = gpioGetMode(PI_ASPI_CE2);
|
|
old_mode_asclk = gpioGetMode(PI_ASPI_SCLK);
|
|
old_mode_amiso = gpioGetMode(PI_ASPI_MISO);
|
|
old_mode_amosi = gpioGetMode(PI_ASPI_MOSI);
|
|
|
|
old_spi_cntl0 = auxReg[AUX_SPI0_CNTL0_REG];
|
|
old_spi_cntl1 = auxReg[AUX_SPI0_CNTL1_REG];
|
|
|
|
/* manually control auxiliary SPI chip selects */
|
|
|
|
if (!(resvd&1))
|
|
{
|
|
myGpioSetMode(PI_ASPI_CE0, PI_OUTPUT);
|
|
myGpioWrite(PI_ASPI_CE0, !(cspols&1));
|
|
}
|
|
|
|
if (!(resvd&2))
|
|
{
|
|
myGpioSetMode(PI_ASPI_CE1, PI_OUTPUT);
|
|
myGpioWrite(PI_ASPI_CE1, !(cspols&2));
|
|
}
|
|
|
|
if (!(resvd&4))
|
|
{
|
|
myGpioSetMode(PI_ASPI_CE2, PI_OUTPUT);
|
|
myGpioWrite(PI_ASPI_CE2, !(cspols&4));
|
|
}
|
|
|
|
/* set gpios to SPI mode */
|
|
|
|
myGpioSetMode(PI_ASPI_SCLK, PI_ALT4);
|
|
myGpioSetMode(PI_ASPI_MISO, PI_ALT4);
|
|
myGpioSetMode(PI_ASPI_MOSI, PI_ALT4);
|
|
}
|
|
else
|
|
{
|
|
/* save original state */
|
|
|
|
old_mode_ce0 = gpioGetMode(PI_SPI_CE0);
|
|
old_mode_ce1 = gpioGetMode(PI_SPI_CE1);
|
|
old_mode_sclk = gpioGetMode(PI_SPI_SCLK);
|
|
old_mode_miso = gpioGetMode(PI_SPI_MISO);
|
|
old_mode_mosi = gpioGetMode(PI_SPI_MOSI);
|
|
|
|
old_spi_cs = spiReg[SPI_CS];
|
|
old_spi_clk = spiReg[SPI_CLK];
|
|
|
|
/* set gpios to SPI mode */
|
|
|
|
if (!(resvd&1)) myGpioSetMode(PI_SPI_CE0, PI_ALT0);
|
|
if (!(resvd&2)) myGpioSetMode(PI_SPI_CE1, PI_ALT0);
|
|
|
|
myGpioSetMode(PI_SPI_SCLK, PI_ALT0);
|
|
myGpioSetMode(PI_SPI_MISO, PI_ALT0);
|
|
myGpioSetMode(PI_SPI_MOSI, PI_ALT0);
|
|
}
|
|
}
|
|
|
|
static void spiTerm(uint32_t flags)
|
|
{
|
|
int resvd;
|
|
|
|
resvd = PI_SPI_FLAGS_GET_RESVD(flags);
|
|
|
|
if (PI_SPI_FLAGS_GET_AUX_SPI(flags))
|
|
{
|
|
/* disable module and access to registers */
|
|
|
|
auxReg[AUX_ENABLES] &= (~AUXENB_SPI1);
|
|
|
|
/* restore original state */
|
|
|
|
if (!(resvd&1)) myGpioSetMode(PI_ASPI_CE0, old_mode_ace0);
|
|
if (!(resvd&2)) myGpioSetMode(PI_ASPI_CE1, old_mode_ace1);
|
|
if (!(resvd&4)) myGpioSetMode(PI_ASPI_CE2, old_mode_ace2);
|
|
|
|
myGpioSetMode(PI_ASPI_SCLK, old_mode_asclk);
|
|
myGpioSetMode(PI_ASPI_MISO, old_mode_amiso);
|
|
myGpioSetMode(PI_ASPI_MOSI, old_mode_amosi);
|
|
|
|
auxReg[AUX_SPI0_CNTL0_REG] = old_spi_cntl0;
|
|
auxReg[AUX_SPI0_CNTL1_REG] = old_spi_cntl1;
|
|
}
|
|
else
|
|
{
|
|
/* restore original state */
|
|
|
|
if (!(resvd&1)) myGpioSetMode(PI_SPI_CE0, old_mode_ce0);
|
|
if (!(resvd&2)) myGpioSetMode(PI_SPI_CE1, old_mode_ce1);
|
|
|
|
myGpioSetMode(PI_SPI_SCLK, old_mode_sclk);
|
|
myGpioSetMode(PI_SPI_MISO, old_mode_miso);
|
|
myGpioSetMode(PI_SPI_MOSI, old_mode_mosi);
|
|
|
|
spiReg[SPI_CS] = old_spi_cs;
|
|
spiReg[SPI_CLK] = old_spi_clk;
|
|
}
|
|
}
|
|
|
|
int spiOpen(unsigned spiChan, unsigned baud, unsigned spiFlags)
|
|
{
|
|
static pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER;
|
|
int i, slot;
|
|
|
|
DBG(DBG_USER, "spiChan=%d baud=%d spiFlags=0x%X",
|
|
spiChan, baud, spiFlags);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (PI_SPI_FLAGS_GET_AUX_SPI(spiFlags))
|
|
{
|
|
if (gpioHardwareRevision() < 16)
|
|
SOFT_ERROR(PI_NO_AUX_SPI, "no auxiliary SPI on Pi A or B");
|
|
|
|
i = PI_NUM_AUX_SPI_CHANNEL;
|
|
}
|
|
else
|
|
i = PI_NUM_STD_SPI_CHANNEL;
|
|
|
|
if (spiChan >= i)
|
|
SOFT_ERROR(PI_BAD_SPI_CHANNEL, "bad spiChan (%d)", spiChan);
|
|
|
|
if ((baud < PI_SPI_MIN_BAUD) || (baud > PI_SPI_MAX_BAUD))
|
|
SOFT_ERROR(PI_BAD_SPI_SPEED, "bad baud (%d)", baud);
|
|
|
|
if (spiFlags > (1<<22))
|
|
SOFT_ERROR(PI_BAD_FLAGS, "bad spiFlags (0x%X)", spiFlags);
|
|
|
|
if (!spiAnyOpen(spiFlags)) /* initialise on first open */
|
|
{
|
|
spiInit(spiFlags);
|
|
spiGo(baud, spiFlags, NULL, NULL, 0);
|
|
}
|
|
|
|
slot = -1;
|
|
|
|
pthread_mutex_lock(&mutex);
|
|
|
|
for (i=0; i<PI_SPI_SLOTS; i++)
|
|
{
|
|
if (spiInfo[i].state == PI_SPI_CLOSED)
|
|
{
|
|
slot = i;
|
|
spiInfo[slot].state = PI_SPI_RESERVED;
|
|
break;
|
|
}
|
|
}
|
|
|
|
pthread_mutex_unlock(&mutex);
|
|
|
|
if (slot < 0) SOFT_ERROR(PI_NO_HANDLE, "no SPI handles");
|
|
|
|
spiInfo[slot].speed = baud;
|
|
spiInfo[slot].flags = spiFlags | PI_SPI_FLAGS_CHANNEL(spiChan);
|
|
spiInfo[slot].state = PI_SPI_OPENED;
|
|
|
|
return slot;
|
|
}
|
|
|
|
int spiClose(unsigned handle)
|
|
{
|
|
DBG(DBG_USER, "handle=%d", handle);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (handle >= PI_SPI_SLOTS)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
if (spiInfo[handle].state != PI_SPI_OPENED)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
spiInfo[handle].state = PI_SPI_CLOSED;
|
|
|
|
if (!spiAnyOpen(spiInfo[handle].flags))
|
|
spiTerm(spiInfo[handle].flags); /* terminate on last close */
|
|
|
|
return 0;
|
|
}
|
|
|
|
int spiRead(unsigned handle, char *buf, unsigned count)
|
|
{
|
|
DBG(DBG_USER, "handle=%d count=%d [%s]",
|
|
handle, count, myBuf2Str(count, buf));
|
|
|
|
CHECK_INITED;
|
|
|
|
if (handle >= PI_SPI_SLOTS)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
if (spiInfo[handle].state != PI_SPI_OPENED)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
if (count > PI_MAX_SPI_DEVICE_COUNT)
|
|
SOFT_ERROR(PI_BAD_SPI_COUNT, "bad count (%d)", count);
|
|
|
|
spiGo(spiInfo[handle].speed, spiInfo[handle].flags, NULL, buf, count);
|
|
|
|
return count;
|
|
}
|
|
|
|
int spiWrite(unsigned handle, char *buf, unsigned count)
|
|
{
|
|
DBG(DBG_USER, "handle=%d count=%d [%s]",
|
|
handle, count, myBuf2Str(count, buf));
|
|
|
|
CHECK_INITED;
|
|
|
|
if (handle >= PI_SPI_SLOTS)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
if (spiInfo[handle].state != PI_SPI_OPENED)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
if (count > PI_MAX_SPI_DEVICE_COUNT)
|
|
SOFT_ERROR(PI_BAD_SPI_COUNT, "bad count (%d)", count);
|
|
|
|
spiGo(spiInfo[handle].speed, spiInfo[handle].flags, buf, NULL, count);
|
|
|
|
return count;
|
|
}
|
|
|
|
int spiXfer(unsigned handle, char *txBuf, char *rxBuf, unsigned count)
|
|
{
|
|
DBG(DBG_USER, "handle=%d count=%d [%s]",
|
|
handle, count, myBuf2Str(count, txBuf));
|
|
|
|
CHECK_INITED;
|
|
|
|
if (handle >= PI_SPI_SLOTS)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
if (spiInfo[handle].state != PI_SPI_OPENED)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
if (count > PI_MAX_SPI_DEVICE_COUNT)
|
|
SOFT_ERROR(PI_BAD_SPI_COUNT, "bad count (%d)", count);
|
|
|
|
spiGo(spiInfo[handle].speed, spiInfo[handle].flags, txBuf, rxBuf, count);
|
|
|
|
return count;
|
|
}
|
|
|
|
/* ======================================================================= */
|
|
|
|
|
|
int serOpen(char *tty, unsigned serBaud, unsigned serFlags)
|
|
{
|
|
static pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER;
|
|
struct termios new;
|
|
int speed;
|
|
int fd;
|
|
int i, slot;
|
|
|
|
DBG(DBG_USER, "tty=%s serBaud=%d serFlags=0x%X", tty, serBaud, serFlags);
|
|
|
|
SER_CHECK_INITED;
|
|
|
|
if (strncmp("/dev/tty", tty, 8) && strncmp("/dev/serial", tty, 11))
|
|
SOFT_ERROR(PI_BAD_SER_DEVICE, "bad device (%s)", tty);
|
|
|
|
switch (serBaud)
|
|
{
|
|
case 50: speed = B50; break;
|
|
case 75: speed = B75; break;
|
|
case 110: speed = B110; break;
|
|
case 134: speed = B134; break;
|
|
case 150: speed = B150; break;
|
|
case 200: speed = B200; break;
|
|
case 300: speed = B300; break;
|
|
case 600: speed = B600; break;
|
|
case 1200: speed = B1200; break;
|
|
case 1800: speed = B1800; break;
|
|
case 2400: speed = B2400; break;
|
|
case 4800: speed = B4800; break;
|
|
case 9600: speed = B9600; break;
|
|
case 19200: speed = B19200; break;
|
|
case 38400: speed = B38400; break;
|
|
case 57600: speed = B57600; break;
|
|
case 115200: speed = B115200; break;
|
|
case 230400: speed = B230400; break;
|
|
|
|
default:
|
|
SOFT_ERROR(PI_BAD_SER_SPEED, "bad speed (%d)", serBaud);
|
|
}
|
|
|
|
if (serFlags)
|
|
SOFT_ERROR(PI_BAD_FLAGS, "bad flags (0x%X)", serFlags);
|
|
|
|
slot = -1;
|
|
|
|
pthread_mutex_lock(&mutex);
|
|
|
|
for (i=0; i<PI_SER_SLOTS; i++)
|
|
{
|
|
if (serInfo[i].state == PI_SER_CLOSED)
|
|
{
|
|
slot = i;
|
|
serInfo[slot].state = PI_SER_RESERVED;
|
|
break;
|
|
}
|
|
}
|
|
|
|
pthread_mutex_unlock(&mutex);
|
|
|
|
if (slot < 0) SOFT_ERROR(PI_NO_HANDLE, "no serial handles");
|
|
|
|
if ((fd = open(tty, O_RDWR | O_NOCTTY | O_NDELAY | O_NONBLOCK)) == -1)
|
|
{
|
|
serInfo[slot].state = PI_SER_CLOSED;
|
|
return PI_SER_OPEN_FAILED;
|
|
}
|
|
|
|
tcgetattr(fd, &new);
|
|
|
|
cfmakeraw(&new);
|
|
|
|
cfsetispeed(&new, speed);
|
|
cfsetospeed(&new, speed);
|
|
|
|
new.c_cc [VMIN] = 0;
|
|
new.c_cc [VTIME] = 0;
|
|
|
|
tcflush(fd, TCIFLUSH);
|
|
tcsetattr(fd, TCSANOW, &new);
|
|
|
|
//fcntl(fd, F_SETFL, O_RDWR);
|
|
|
|
serInfo[slot].fd = fd;
|
|
serInfo[slot].flags = serFlags;
|
|
serInfo[slot].state = PI_SER_OPENED;
|
|
|
|
return slot;
|
|
}
|
|
|
|
int serClose(unsigned handle)
|
|
{
|
|
DBG(DBG_USER, "handle=%d", handle);
|
|
|
|
SER_CHECK_INITED;
|
|
|
|
if (handle >= PI_SER_SLOTS)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
if (serInfo[handle].state != PI_SER_OPENED)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
if (serInfo[handle].fd >= 0) close(serInfo[handle].fd);
|
|
|
|
serInfo[handle].fd = -1;
|
|
serInfo[handle].state = PI_SER_CLOSED;
|
|
|
|
return 0;
|
|
}
|
|
|
|
int serWriteByte(unsigned handle, unsigned bVal)
|
|
{
|
|
char c;
|
|
|
|
DBG(DBG_USER, "handle=%d bVal=%d", handle, bVal);
|
|
|
|
SER_CHECK_INITED;
|
|
|
|
if (handle >= PI_SER_SLOTS)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
if (serInfo[handle].state != PI_SER_OPENED)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
if (bVal > 0xFF)
|
|
SOFT_ERROR(PI_BAD_PARAM, "bad parameter (%d)", bVal);
|
|
|
|
c = bVal;
|
|
|
|
if (write(serInfo[handle].fd, &c, 1) != 1)
|
|
return PI_SER_WRITE_FAILED;
|
|
else
|
|
return 0;
|
|
}
|
|
|
|
int serReadByte(unsigned handle)
|
|
{
|
|
int r;
|
|
char x;
|
|
|
|
DBG(DBG_USER, "handle=%d", handle);
|
|
|
|
SER_CHECK_INITED;
|
|
|
|
if (handle >= PI_SER_SLOTS)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
if (serInfo[handle].state != PI_SER_OPENED)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
r = read(serInfo[handle].fd, &x, 1);
|
|
|
|
if (r == 1)
|
|
return ((int)x) & 0xFF;
|
|
|
|
else if (r == 0)
|
|
return PI_SER_READ_NO_DATA;
|
|
|
|
else if ((r == -1) && (errno == EAGAIN))
|
|
return PI_SER_READ_NO_DATA;
|
|
|
|
else
|
|
return PI_SER_READ_FAILED;
|
|
}
|
|
|
|
int serWrite(unsigned handle, char *buf, unsigned count)
|
|
{
|
|
int written=0, wrote=0;
|
|
|
|
DBG(DBG_USER, "handle=%d count=%d [%s]",
|
|
handle, count, myBuf2Str(count, buf));
|
|
|
|
SER_CHECK_INITED;
|
|
|
|
if (handle >= PI_SER_SLOTS)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
if (serInfo[handle].state != PI_SER_OPENED)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
if (!count)
|
|
SOFT_ERROR(PI_BAD_PARAM, "bad count (%d)", count);
|
|
|
|
while ((written != count) && (wrote >= 0))
|
|
{
|
|
wrote = write(serInfo[handle].fd, buf+written, count-written);
|
|
|
|
if (wrote >= 0)
|
|
{
|
|
written += wrote;
|
|
|
|
if (written != count) time_sleep(0.05);
|
|
}
|
|
}
|
|
|
|
if (written != count)
|
|
return PI_SER_WRITE_FAILED;
|
|
else
|
|
return 0;
|
|
}
|
|
|
|
int serRead(unsigned handle, char *buf, unsigned count)
|
|
{
|
|
int r;
|
|
|
|
DBG(DBG_USER, "handle=%d count=%d buf=0x%"PRIXPTR, handle, count, (uintptr_t)buf);
|
|
|
|
SER_CHECK_INITED;
|
|
|
|
if (handle >= PI_SER_SLOTS)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
if (serInfo[handle].state != PI_SER_OPENED)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
if (!count)
|
|
SOFT_ERROR(PI_BAD_PARAM, "bad count (%d)", count);
|
|
|
|
r = read(serInfo[handle].fd, buf, count);
|
|
|
|
if (r == -1)
|
|
{
|
|
if (errno == EAGAIN)
|
|
return PI_SER_READ_NO_DATA;
|
|
else
|
|
return PI_SER_READ_FAILED;
|
|
}
|
|
else
|
|
{
|
|
if (r < count) buf[r] = 0;
|
|
return r;
|
|
}
|
|
}
|
|
|
|
int serDataAvailable(unsigned handle)
|
|
{
|
|
int result;
|
|
|
|
DBG(DBG_USER, "handle=%d", handle);
|
|
|
|
SER_CHECK_INITED;
|
|
|
|
if (handle >= PI_SER_SLOTS)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
if (serInfo[handle].state != PI_SER_OPENED)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
if (ioctl(serInfo[handle].fd, FIONREAD, &result) == -1) return 0;
|
|
|
|
return result;
|
|
}
|
|
|
|
/* ======================================================================= */
|
|
|
|
static int chooseBestClock
|
|
(clkInf_t *clkInf, unsigned f, unsigned numc, unsigned *cf)
|
|
{
|
|
int c, valid;
|
|
double fdiv, offby, best_offby;
|
|
unsigned div, frac;
|
|
|
|
valid = 0;
|
|
best_offby = 0;
|
|
|
|
for (c=0; c<numc; c++)
|
|
{
|
|
fdiv = (double)cf[c] / (double)f;
|
|
if (f < PI_MASH_MAX_FREQ)
|
|
{
|
|
fdiv += (0.5 / 4096.0);
|
|
div = fdiv;
|
|
frac = (fdiv - div) * 4096.0;
|
|
}
|
|
else
|
|
{
|
|
fdiv += 0.5;
|
|
div = fdiv;
|
|
frac = 0;
|
|
}
|
|
|
|
if ((div > 1) && (div < 4096))
|
|
{
|
|
offby = f - (cf[c] / (div + (frac / 4096.0)));
|
|
if (offby < 0) offby = - offby;
|
|
if ((!valid) || (offby <= best_offby))
|
|
{
|
|
valid = 1;
|
|
clkInf->div = div;
|
|
clkInf->frac = frac;
|
|
clkInf->clock = c;
|
|
best_offby = offby;
|
|
}
|
|
}
|
|
}
|
|
return valid;
|
|
}
|
|
|
|
/* ======================================================================= */
|
|
|
|
static rawCbs_t * dmaCB2adr(int pos)
|
|
{
|
|
int page, slot;
|
|
|
|
page = pos/CBS_PER_IPAGE;
|
|
slot = pos%CBS_PER_IPAGE;
|
|
|
|
return &dmaIVirt[page]->cb[slot];
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static void dmaCbPrint(int pos)
|
|
{
|
|
rawCbs_t * p;
|
|
|
|
p = dmaCB2adr(pos);
|
|
|
|
fprintf(stderr, "i=%x s=%x d=%x len=%x s=%x nxt=%x\n",
|
|
p->info, p->src, p->dst, p->length, p->stride, p->next);
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static unsigned dmaNowAtICB(void)
|
|
{
|
|
unsigned cb;
|
|
static unsigned lastPage=0;
|
|
unsigned page;
|
|
uint32_t cbAddr;
|
|
uint32_t startTick, endTick;
|
|
|
|
startTick = systReg[SYST_CLO];
|
|
|
|
cbAddr = dmaIn[DMA_CONBLK_AD];
|
|
|
|
page = lastPage;
|
|
|
|
/* which page are we dma'ing? */
|
|
|
|
while (1)
|
|
{
|
|
//cast twice to suppress compiler warning, I belive this cast is ok
|
|
//because dmaIbus contains bus addresses, not user addresses. --plugwash
|
|
cb = (cbAddr - ((int)(uintptr_t)dmaIBus[page])) / 32;
|
|
|
|
if (cb < CBS_PER_IPAGE)
|
|
{
|
|
endTick = systReg[SYST_CLO];
|
|
|
|
if (endTick != startTick)
|
|
gpioStats.cbTicks += (endTick - startTick);
|
|
|
|
gpioStats.cbCalls++;
|
|
|
|
lastPage = page;
|
|
|
|
return (page*CBS_PER_IPAGE) + cb;
|
|
}
|
|
|
|
if (page++ >= DMAI_PAGES) page=0;
|
|
|
|
if (page == lastPage) break;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static int dmaNowAtOCB(void)
|
|
{
|
|
unsigned cb;
|
|
unsigned page;
|
|
uint32_t cbAddr;
|
|
|
|
cbAddr = dmaOut[DMA_CONBLK_AD];
|
|
|
|
if (!cbAddr) return -PI_NO_TX_WAVE;
|
|
|
|
page = 0;
|
|
|
|
/* which page are we dma'ing? */
|
|
|
|
while (1)
|
|
{
|
|
//cast twice to suppress compiler warning, I belive this cast is ok
|
|
//because dmaIbus contains bus addresses, not user addresses. --plugwash
|
|
cb = (cbAddr - ((int)(uintptr_t)dmaOBus[page])) / 32;
|
|
|
|
if (cb < CBS_PER_OPAGE) return (page*CBS_PER_OPAGE) + cb;
|
|
|
|
if (page++ >= DMAO_PAGES) break;
|
|
}
|
|
|
|
/* Try twice */
|
|
|
|
cbAddr = dmaOut[DMA_CONBLK_AD];
|
|
|
|
if (!cbAddr) return -PI_NO_TX_WAVE;
|
|
|
|
page = 0;
|
|
|
|
/* which page are we dma'ing? */
|
|
|
|
while (1)
|
|
{
|
|
//cast twice to suppress compiler warning, I belive this cast is ok
|
|
//because dmaIbus contains bus addresses, not user addresses. --plugwash
|
|
cb = (cbAddr - ((int)(uintptr_t)dmaOBus[page])) / 32;
|
|
|
|
if (cb < CBS_PER_OPAGE) return (page*CBS_PER_OPAGE) + cb;
|
|
|
|
if (page++ >= DMAO_PAGES) break;
|
|
}
|
|
|
|
return -PI_WAVE_NOT_FOUND;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
unsigned rawWaveCB(void)
|
|
{
|
|
unsigned cb;
|
|
static unsigned lastPage=0;
|
|
unsigned page;
|
|
uint32_t cbAddr;
|
|
|
|
cbAddr = dmaOut[DMA_CONBLK_AD];
|
|
|
|
if (!cbAddr) return -1;
|
|
|
|
page = lastPage;
|
|
|
|
/* which page are we dma'ing? */
|
|
|
|
while (1)
|
|
{
|
|
//cast twice to suppress compiler warning, I belive this cast is ok
|
|
//because dmaIbus contains bus addresses, not user addresses. --plugwash
|
|
cb = (cbAddr - ((int)(uintptr_t)dmaOBus[page])) / 32;
|
|
|
|
if (cb < CBS_PER_OPAGE)
|
|
{
|
|
lastPage = page;
|
|
|
|
return (page*CBS_PER_OPAGE) + cb;
|
|
}
|
|
|
|
if (page++ >= DMAO_PAGES) page=0;
|
|
|
|
if (page == lastPage) break;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static unsigned dmaCurrentSlot(unsigned pos)
|
|
{
|
|
unsigned cycle=0, slot=0, tmp;
|
|
|
|
cycle = (pos/CBS_PER_CYCLE);
|
|
tmp = (pos%CBS_PER_CYCLE);
|
|
|
|
if (tmp > 2) slot = ((tmp-2)/3);
|
|
|
|
return (cycle*PULSE_PER_CYCLE)+slot;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static uint32_t dmaPwmDataAdr(int pos)
|
|
{
|
|
//cast twice to suppress compiler warning, I belive this cast is ok
|
|
//because dmaIbus contains bus addresses, not user addresses. --plugwash
|
|
return (uint32_t)(uintptr_t) &dmaIBus[pos]->periphData;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static uint32_t dmaGpioOnAdr(int pos)
|
|
{
|
|
int page, slot;
|
|
|
|
page = pos/ON_PER_IPAGE;
|
|
slot = pos%ON_PER_IPAGE;
|
|
|
|
//cast twice to suppress compiler warning, I belive this cast is ok
|
|
//because dmaIbus contains bus addresses, not user addresses. --plugwash
|
|
return (uint32_t)(uintptr_t) &dmaIBus[page]->gpioOn[slot];
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static uint32_t dmaGpioOffAdr(int pos)
|
|
{
|
|
int page, slot;
|
|
|
|
myOffPageSlot(pos, &page, &slot);
|
|
|
|
//cast twice to suppress compiler warning, I belive this cast is ok
|
|
//because dmaIbus contains bus addresses, not user addresses. --plugwash
|
|
return (uint32_t)(uintptr_t) &dmaIBus[page]->gpioOff[slot];
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static uint32_t dmaTickAdr(int pos)
|
|
{
|
|
int page, slot;
|
|
|
|
myTckPageSlot(pos, &page, &slot);
|
|
|
|
//cast twice to suppress compiler warning, I belive this cast is ok
|
|
//because dmaIbus contains bus addresses, not user addresses. --plugwash
|
|
return (uint32_t)(uintptr_t) &dmaIBus[page]->tick[slot];
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static uint32_t dmaReadLevelsAdr(int pos)
|
|
{
|
|
int page, slot;
|
|
|
|
myLvsPageSlot(pos, &page, &slot);
|
|
|
|
//cast twice to suppress compiler warning, I belive this cast is ok
|
|
//because dmaIbus contains bus addresses, not user addresses. --plugwash
|
|
return (uint32_t)(uintptr_t) &dmaIBus[page]->level[slot];
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static uint32_t dmaCbAdr(int pos)
|
|
{
|
|
int page, slot;
|
|
|
|
page = (pos/CBS_PER_IPAGE);
|
|
slot = (pos%CBS_PER_IPAGE);
|
|
|
|
//cast twice to suppress compiler warning, I belive this cast is ok
|
|
//because dmaIbus contains bus addresses, not user addresses. --plugwash
|
|
return (uint32_t)(uintptr_t) &dmaIBus[page]->cb[slot];
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static void dmaGpioOnCb(int b, int pos)
|
|
{
|
|
rawCbs_t * p;
|
|
|
|
p = dmaCB2adr(b);
|
|
|
|
p->info = NORMAL_DMA;
|
|
p->src = dmaGpioOnAdr(pos);
|
|
p->dst = ((GPIO_BASE + (GPSET0*4)) & 0x00ffffff) | PI_PERI_BUS;
|
|
p->length = 4;
|
|
p->next = dmaCbAdr(b+1);
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static void dmaTickCb(int b, int pos)
|
|
{
|
|
rawCbs_t * p;
|
|
|
|
p = dmaCB2adr(b);
|
|
|
|
p->info = NORMAL_DMA;
|
|
p->src = ((SYST_BASE + (SYST_CLO*4)) & 0x00ffffff) | PI_PERI_BUS;
|
|
p->dst = dmaTickAdr(pos);
|
|
p->length = 4;
|
|
p->next = dmaCbAdr(b+1);
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static void dmaGpioOffCb(int b, int pos)
|
|
{
|
|
rawCbs_t * p;
|
|
|
|
p = dmaCB2adr(b);
|
|
|
|
p->info = NORMAL_DMA;
|
|
p->src = dmaGpioOffAdr(pos);
|
|
p->dst = ((GPIO_BASE + (GPCLR0*4)) & 0x00ffffff) | PI_PERI_BUS;
|
|
p->length = 4;
|
|
p->next = dmaCbAdr(b+1);
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static void dmaReadLevelsCb(int b, int pos)
|
|
{
|
|
rawCbs_t * p;
|
|
|
|
p = dmaCB2adr(b);
|
|
|
|
p->info = NORMAL_DMA;
|
|
p->src = ((GPIO_BASE + (GPLEV0*4)) & 0x00ffffff) | PI_PERI_BUS;
|
|
p->dst = dmaReadLevelsAdr(pos);
|
|
p->length = 4;
|
|
p->next = dmaCbAdr(b+1);
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static void dmaDelayCb(int b)
|
|
{
|
|
rawCbs_t * p;
|
|
|
|
p = dmaCB2adr(b);
|
|
|
|
if (gpioCfg.clockPeriph == PI_CLOCK_PCM)
|
|
{
|
|
p->info = NORMAL_DMA | TIMED_DMA(2);
|
|
p->dst = PCM_TIMER;
|
|
}
|
|
else
|
|
{
|
|
p->info = NORMAL_DMA | TIMED_DMA(5);
|
|
p->dst = PWM_TIMER;
|
|
}
|
|
|
|
p->src = dmaPwmDataAdr(b%DMAI_PAGES);
|
|
p->length = 4;
|
|
p->next = dmaCbAdr(b+1);
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static void dmaInitCbs(void)
|
|
{
|
|
int b, pulse, level, cycle;
|
|
|
|
rawCbs_t * p;
|
|
|
|
/* set up the DMA control blocks */
|
|
|
|
DBG(DBG_STARTUP, "");
|
|
|
|
gpioStats.dmaInitCbsCount++;
|
|
|
|
b = -1;
|
|
level = 0;
|
|
|
|
for (cycle=0; cycle<bufferCycles; cycle++)
|
|
{
|
|
b++; dmaGpioOnCb(b, cycle%SUPERCYCLE); /* gpio on slot */
|
|
|
|
b++; dmaTickCb(b, cycle); /* tick slot */
|
|
|
|
for (pulse=0; pulse<PULSE_PER_CYCLE; pulse++)
|
|
{
|
|
b++; dmaReadLevelsCb(b, level); /* read levels slot */
|
|
|
|
b++; dmaDelayCb(b); /* delay slot */
|
|
|
|
b++; dmaGpioOffCb(b, (level%SUPERLEVEL)+1); /* gpio off slot */
|
|
|
|
++level;
|
|
}
|
|
}
|
|
|
|
/* point last cb back to first for continuous loop */
|
|
|
|
p = dmaCB2adr(b);
|
|
|
|
p->next = dmaCbAdr(0);
|
|
|
|
DBG(DBG_STARTUP, "DMA page type count = %zd", sizeof(dmaIPage_t));
|
|
|
|
DBG(DBG_STARTUP, "%d control blocks (exp=%d)", b+1, NUM_CBS);
|
|
}
|
|
|
|
/* ======================================================================= */
|
|
|
|
|
|
static void sigHandler(int signum)
|
|
{
|
|
if ((signum >= PI_MIN_SIGNUM) && (signum <= PI_MAX_SIGNUM))
|
|
{
|
|
if (gpioSignal[signum].func)
|
|
{
|
|
if (gpioSignal[signum].ex)
|
|
{
|
|
(gpioSignal[signum].func)(signum, gpioSignal[signum].userdata);
|
|
}
|
|
else
|
|
{
|
|
(gpioSignal[signum].func)(signum);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
switch(signum)
|
|
{
|
|
case SIGUSR1:
|
|
|
|
if (gpioCfg.dbgLevel > DBG_MIN_LEVEL) --gpioCfg.dbgLevel;
|
|
else gpioCfg.dbgLevel = DBG_MIN_LEVEL;
|
|
DBG(DBG_USER, "Debug level %d\n", gpioCfg.dbgLevel);
|
|
break;
|
|
|
|
case SIGUSR2:
|
|
if (gpioCfg.dbgLevel < DBG_MAX_LEVEL) ++gpioCfg.dbgLevel;
|
|
else gpioCfg.dbgLevel = DBG_MAX_LEVEL;
|
|
DBG(DBG_USER, "Debug level %d\n", gpioCfg.dbgLevel);
|
|
break;
|
|
|
|
case SIGPIPE:
|
|
case SIGWINCH:
|
|
DBG(DBG_USER, "signal %d ignored", signum);
|
|
break;
|
|
|
|
case SIGCHLD:
|
|
/* Used to notify threads of events */
|
|
break;
|
|
|
|
default:
|
|
DBG(DBG_ALWAYS, "Unhandled signal %d, terminating\n", signum);
|
|
gpioTerminate();
|
|
exit(-1);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
/* exit */
|
|
|
|
DBG(DBG_ALWAYS, "Unhandled signal %d, terminating\n", signum);
|
|
gpioTerminate();
|
|
exit(-1);
|
|
}
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static void sigSetHandler(void)
|
|
{
|
|
int i;
|
|
struct sigaction new;
|
|
|
|
for (i=PI_MIN_SIGNUM; i<=PI_MAX_SIGNUM; i++)
|
|
{
|
|
|
|
memset(&new, 0, sizeof(new));
|
|
new.sa_handler = sigHandler;
|
|
|
|
sigaction(i, &new, NULL);
|
|
}
|
|
}
|
|
|
|
/*
|
|
freq mics net
|
|
0 1000 1000 900
|
|
1 4000 250 225
|
|
2 3750 266 240
|
|
3 3500 285 257
|
|
4 3250 307 276
|
|
5 3000 333 300
|
|
6 2750 363 327
|
|
7 2500 400 360
|
|
8 2250 444 400
|
|
9 2000 500 450
|
|
10 1750 571 514
|
|
11 1500 666 600
|
|
12 1250 800 720
|
|
13 1000 1000 900
|
|
14 750 1333 1200
|
|
15 500 2000 1800
|
|
*/
|
|
|
|
unsigned alert_delays[]=
|
|
{
|
|
900000, 225000, 240000, 257142, 276923, 300000, 327272, 360000,
|
|
400000, 450000, 514285, 600000, 720000, 900000, 1200000, 1800000
|
|
};
|
|
|
|
/* ======================================================================= */
|
|
|
|
static void alertGlitchFilter(gpioSample_t *sample, int numSamples)
|
|
{
|
|
int i, j, diff;
|
|
uint32_t steadyUs, changedTick, RBitV, LBitV, initialised;
|
|
uint32_t bit, bitV;
|
|
|
|
for (i=0; i<=PI_MAX_USER_GPIO; i++)
|
|
{
|
|
bit = (1<<i);
|
|
|
|
if (monitorBits & bit & gFilterBits)
|
|
{
|
|
initialised = gpioAlert[i].gfInitialised;
|
|
if (!initialised && numSamples > 0)
|
|
{
|
|
/* Initialise filter with first sample */
|
|
bitV = sample[0].level & bit;
|
|
gpioAlert[i].gfRBitV = bitV;
|
|
gpioAlert[i].gfLBitV = bitV;
|
|
gpioAlert[i].gfTick = sample[0].tick;
|
|
gpioAlert[i].gfInitialised = 1;
|
|
}
|
|
|
|
steadyUs = gpioAlert[i].gfSteadyUs;
|
|
RBitV = gpioAlert[i].gfRBitV;
|
|
LBitV = gpioAlert[i].gfLBitV;
|
|
changedTick = gpioAlert[i].gfTick;
|
|
|
|
for (j=0; j<numSamples; j++)
|
|
{
|
|
bitV = sample[j].level & bit;
|
|
|
|
if (bitV != LBitV)
|
|
{
|
|
/* Difference between level and last level.
|
|
Restart steady timer. */
|
|
|
|
changedTick = sample[j].tick;
|
|
LBitV = bitV;
|
|
}
|
|
|
|
if (bitV != RBitV)
|
|
{
|
|
/* Difference between level and reported level. */
|
|
|
|
diff = sample[j].tick - changedTick;
|
|
|
|
if (diff >= steadyUs)
|
|
{
|
|
/* Level stable for steady period. */
|
|
RBitV = bitV;
|
|
}
|
|
else
|
|
{
|
|
/* Keep reporting old level. */
|
|
|
|
sample[j].level ^= bit;
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
gpioAlert[i].gfRBitV = RBitV;
|
|
gpioAlert[i].gfLBitV = LBitV;
|
|
gpioAlert[i].gfTick = changedTick;
|
|
}
|
|
}
|
|
}
|
|
|
|
static void alertNoiseFilter(gpioSample_t *sample, int numSamples)
|
|
{
|
|
int i, j, diff;
|
|
uint32_t LBitV;
|
|
uint32_t bit, bitV;
|
|
uint32_t nowTick;
|
|
|
|
for (i=0; i<=PI_MAX_USER_GPIO; i++)
|
|
{
|
|
bit = (1<<i);
|
|
|
|
if (monitorBits & bit & nFilterBits)
|
|
{
|
|
LBitV = gpioAlert[i].nfLBitV;
|
|
|
|
for (j=0; j<numSamples; j++)
|
|
{
|
|
bitV = sample[j].level & bit;
|
|
nowTick = sample[j].tick;
|
|
|
|
if (gpioAlert[i].nfActive) /* reporting events */
|
|
{
|
|
diff = nowTick - gpioAlert[i].nfTick2;
|
|
|
|
if (diff >= 0)
|
|
{
|
|
/* Stop reporting gpio changes */
|
|
|
|
gpioAlert[i].nfActive = 0;
|
|
gpioAlert[i].nfTick1 = nowTick;
|
|
}
|
|
}
|
|
else /* waiting for steady us */
|
|
{
|
|
if (bitV != LBitV)
|
|
{
|
|
diff = nowTick - gpioAlert[i].nfTick1;
|
|
gpioAlert[i].nfTick1 = nowTick;
|
|
|
|
if (diff >= gpioAlert[i].nfSteadyUs)
|
|
{
|
|
/* Start reporting gpio changes */
|
|
|
|
gpioAlert[i].nfRBitV = LBitV;
|
|
gpioAlert[i].nfActive = 1;
|
|
gpioAlert[i].nfTick2 =
|
|
nowTick + gpioAlert[i].nfActiveUs;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!gpioAlert[i].nfActive)
|
|
{
|
|
if (bitV != gpioAlert[i].nfRBitV)
|
|
sample[j].level ^= bit;
|
|
}
|
|
|
|
LBitV = bitV;
|
|
}
|
|
|
|
gpioAlert[i].nfLBitV = LBitV;
|
|
|
|
}
|
|
}
|
|
}
|
|
|
|
static void alertEmit(
|
|
gpioSample_t *sample, int numSamples, uint32_t changedBits, uint32_t eTick)
|
|
{
|
|
uint32_t oldLevel, newLevel;
|
|
int32_t diff;
|
|
int emit, seqno, emitted;
|
|
uint32_t changes, bits, timeoutBits, eventBits;
|
|
int d;
|
|
int b, n, v;
|
|
int err;
|
|
int max_emits;
|
|
char fifo[32];
|
|
/* ensure space for maximum number of watchdog and event notifications */
|
|
gpioReport_t report[MAX_REPORT+PI_MAX_USER_GPIO+1+PI_MAX_EVENT+1];
|
|
|
|
if (changedBits)
|
|
{
|
|
if (gpioGetSamples.func)
|
|
{
|
|
if (gpioGetSamples.ex)
|
|
{
|
|
(gpioGetSamples.func)
|
|
(sample, numSamples, gpioGetSamples.userdata);
|
|
}
|
|
else
|
|
{
|
|
(gpioGetSamples.func)(sample, numSamples);
|
|
}
|
|
}
|
|
}
|
|
|
|
eventBits = 0;
|
|
|
|
if (bscFR != (bscsReg[BSC_FR]&0xffff))
|
|
{
|
|
bscFR = bscsReg[BSC_FR]&0xffff;
|
|
eventAlert[PI_EVENT_BSC].fired = 1;
|
|
}
|
|
|
|
for (b=0; b<=PI_MAX_EVENT; b++)
|
|
{
|
|
if (eventAlert[b].fired && (!eventAlert[b].ignore))
|
|
{
|
|
eventBits |= (1<<b);
|
|
|
|
if (eventAlert[b].func)
|
|
{
|
|
if (eventAlert[b].ex)
|
|
{
|
|
(eventAlert[b].func)(b, eTick, eventAlert[b].userdata);
|
|
}
|
|
else
|
|
{
|
|
(eventAlert[b].func)(b, eTick);
|
|
}
|
|
}
|
|
}
|
|
|
|
eventAlert[b].fired = 0;
|
|
}
|
|
|
|
/* call alert callbacks for each bit transition */
|
|
|
|
if (changedBits & alertBits)
|
|
{
|
|
oldLevel = (reportedLevel & alertBits);
|
|
|
|
for (d=0; d<numSamples; d++)
|
|
{
|
|
newLevel = (sample[d].level & alertBits);
|
|
|
|
if (newLevel != oldLevel)
|
|
{
|
|
changes = (newLevel ^ oldLevel);
|
|
|
|
for (b=0; b<=PI_MAX_USER_GPIO; b++)
|
|
{
|
|
if (changes & (1<<b))
|
|
{
|
|
if (newLevel & (1<<b)) v = 1; else v = 0;
|
|
|
|
if (gpioAlert[b].func)
|
|
{
|
|
if (gpioAlert[b].ex)
|
|
{
|
|
(gpioAlert[b].func)
|
|
(b, v, sample[d].tick,
|
|
gpioAlert[b].userdata);
|
|
}
|
|
else
|
|
{
|
|
(gpioAlert[b].func)(b, v, sample[d].tick);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
oldLevel = newLevel;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* check for watchdog timeouts */
|
|
|
|
timeoutBits = 0;
|
|
|
|
if (wdogBits)
|
|
{
|
|
for (b=0; b<=PI_MAX_USER_GPIO; b++)
|
|
{
|
|
if (gpioAlert[b].wdSteadyUs)
|
|
{
|
|
diff = eTick - gpioAlert[b].wdTick;
|
|
|
|
if (diff >= gpioAlert[b].wdSteadyUs)
|
|
{
|
|
timeoutBits |= (1<<b);
|
|
|
|
gpioAlert[b].wdTick = eTick;
|
|
|
|
if (gpioAlert[b].func)
|
|
{
|
|
if (gpioAlert[b].ex)
|
|
{
|
|
(gpioAlert[b].func)(b, PI_TIMEOUT, eTick,
|
|
gpioAlert[b].userdata);
|
|
}
|
|
else
|
|
{
|
|
(gpioAlert[b].func)(b, PI_TIMEOUT, eTick);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
for (n=0; n<PI_NOTIFY_SLOTS; n++)
|
|
{
|
|
if (gpioNotify[n].state == PI_NOTIFY_CLOSING)
|
|
{
|
|
if (gpioNotify[n].pipe)
|
|
{
|
|
DBG(DBG_INTERNAL, "close notify pipe %d", gpioNotify[n].fd);
|
|
close(gpioNotify[n].fd);
|
|
|
|
sprintf(fifo, "/dev/pigpio%d", n);
|
|
|
|
unlink(fifo);
|
|
}
|
|
|
|
gpioNotify[n].state = PI_NOTIFY_CLOSED;
|
|
}
|
|
else if (gpioNotify[n].state >= PI_NOTIFY_OPENED)
|
|
{
|
|
bits = gpioNotify[n].bits;
|
|
|
|
emit = 0;
|
|
|
|
seqno = gpioNotify[n].seqno;
|
|
|
|
if (gpioNotify[n].state == PI_NOTIFY_RUNNING)
|
|
{
|
|
/* check to see if any bits have changed for this
|
|
notification.
|
|
|
|
bits is the set of notification bits
|
|
changedBits is the set of changed bits
|
|
*/
|
|
|
|
if (changedBits & bits)
|
|
{
|
|
oldLevel = reportedLevel & bits;
|
|
|
|
for (d=0; d<numSamples; d++)
|
|
{
|
|
newLevel = sample[d].level & bits;
|
|
|
|
if (newLevel != oldLevel)
|
|
{
|
|
report[emit].seqno = seqno;
|
|
report[emit].flags = 0;
|
|
report[emit].tick = sample[d].tick;
|
|
report[emit].level = sample[d].level;
|
|
|
|
oldLevel = newLevel;
|
|
|
|
emit++;
|
|
seqno++;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* check to see if any watchdogs are due for this
|
|
notification.
|
|
|
|
bits is the set of notification bits
|
|
timeoutBits is the set of timed out bits
|
|
*/
|
|
|
|
bits = gpioNotify[n].bits;
|
|
|
|
if (timeoutBits & bits)
|
|
{
|
|
/* at least one watchdog has fired for this
|
|
notification.
|
|
*/
|
|
|
|
for (b=0; b<=PI_MAX_USER_GPIO; b++)
|
|
{
|
|
if (timeoutBits & bits & (1<<b))
|
|
{
|
|
if (numSamples)
|
|
newLevel = sample[numSamples-1].level;
|
|
else
|
|
newLevel = reportedLevel;
|
|
|
|
report[emit].seqno = seqno;
|
|
report[emit].flags =
|
|
PI_NTFY_FLAGS_WDOG | PI_NTFY_FLAGS_BIT(b);
|
|
report[emit].tick = eTick;
|
|
report[emit].level = newLevel;
|
|
|
|
emit++;
|
|
seqno++;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/* check to see if any events are due
|
|
|
|
eventBits is the set of events
|
|
*/
|
|
|
|
if (eventBits & gpioNotify[n].eventBits)
|
|
{
|
|
for (b=0; b<=PI_MAX_EVENT; b++)
|
|
{
|
|
if (eventBits & gpioNotify[n].eventBits & (1<<b))
|
|
{
|
|
if (numSamples)
|
|
newLevel = sample[numSamples-1].level;
|
|
else
|
|
newLevel = reportedLevel;
|
|
|
|
report[emit].seqno = seqno;
|
|
report[emit].flags =
|
|
PI_NTFY_FLAGS_EVENT | PI_NTFY_FLAGS_BIT(b);
|
|
report[emit].tick = eTick;
|
|
report[emit].level = newLevel;
|
|
|
|
emit++;
|
|
seqno++;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!emit)
|
|
{
|
|
if ((int)(eTick - gpioNotify[n].lastReportTick) > 60000000)
|
|
{
|
|
if (numSamples)
|
|
newLevel = sample[numSamples-1].level;
|
|
else
|
|
newLevel = reportedLevel;
|
|
|
|
report[emit].seqno = seqno;
|
|
report[emit].flags = PI_NTFY_FLAGS_ALIVE;
|
|
report[emit].tick = eTick;
|
|
report[emit].level = newLevel;
|
|
|
|
emit++;
|
|
seqno++;
|
|
}
|
|
}
|
|
|
|
if (emit)
|
|
{
|
|
DBG(DBG_FAST_TICK, "notification %d (%d reports, %x-%x)",
|
|
n, emit, report[0].seqno, report[emit-1].seqno);
|
|
gpioNotify[n].lastReportTick = eTick;
|
|
max_emits = gpioNotify[n].max_emits;
|
|
|
|
if (emit > gpioStats.maxEmit) gpioStats.maxEmit = emit;
|
|
|
|
emitted = 0;
|
|
|
|
while (emit > 0)
|
|
{
|
|
if (emit > max_emits)
|
|
{
|
|
gpioStats.emitFrags++;
|
|
|
|
err = write(gpioNotify[n].fd,
|
|
report+emitted,
|
|
max_emits*sizeof(gpioReport_t));
|
|
|
|
if (err != (max_emits*sizeof(gpioReport_t)))
|
|
{
|
|
if (err < 0)
|
|
{
|
|
if ((errno != EAGAIN) && (errno != EWOULDBLOCK))
|
|
{
|
|
/* serious error, no point continuing */
|
|
|
|
DBG(DBG_ALWAYS, "fd=%d err=%d errno=%d",
|
|
gpioNotify[n].fd, err, errno);
|
|
|
|
DBG(DBG_ALWAYS, "%s", strerror(errno));
|
|
|
|
gpioNotify[n].bits = 0;
|
|
gpioNotify[n].state = PI_NOTIFY_CLOSING;
|
|
intNotifyBits();
|
|
break;
|
|
}
|
|
else gpioStats.wouldBlockPipeWrite++;
|
|
}
|
|
else
|
|
{
|
|
gpioStats.shortPipeWrite++;
|
|
DBG(DBG_ALWAYS, "emitted %zd, asked for %d",
|
|
err/sizeof(gpioReport_t), max_emits);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
gpioStats.goodPipeWrite++;
|
|
}
|
|
|
|
emitted += max_emits;
|
|
emit -= max_emits;
|
|
}
|
|
else
|
|
{
|
|
err = write(gpioNotify[n].fd,
|
|
report+emitted,
|
|
emit*sizeof(gpioReport_t));
|
|
|
|
if (err != (emit*sizeof(gpioReport_t)))
|
|
{
|
|
if (err < 0)
|
|
{
|
|
if ((errno != EAGAIN) && (errno != EWOULDBLOCK))
|
|
{
|
|
DBG(DBG_ALWAYS, "fd=%d err=%d errno=%d",
|
|
gpioNotify[n].fd, err, errno);
|
|
|
|
DBG(DBG_ALWAYS, "%s", strerror(errno));
|
|
|
|
/* serious error, no point continuing */
|
|
gpioNotify[n].bits = 0;
|
|
gpioNotify[n].state = PI_NOTIFY_CLOSING;
|
|
intNotifyBits();
|
|
break;
|
|
}
|
|
else gpioStats.wouldBlockPipeWrite++;
|
|
}
|
|
else
|
|
{
|
|
gpioStats.shortPipeWrite++;
|
|
DBG(DBG_ALWAYS, "emitted %zd, asked for %d",
|
|
err/sizeof(gpioReport_t), emit);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
gpioStats.goodPipeWrite++;
|
|
}
|
|
|
|
emitted += emit;
|
|
emit = 0;
|
|
}
|
|
}
|
|
|
|
gpioNotify[n].seqno = seqno;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (changedBits & scriptBits)
|
|
{
|
|
for (n=0; n<PI_MAX_SCRIPTS; n++)
|
|
{
|
|
if ((gpioScript[n].state == PI_SCRIPT_IN_USE) &&
|
|
(gpioScript[n].run_state == PI_SCRIPT_WAITING) &&
|
|
(gpioScript[n].waitBits & changedBits))
|
|
{
|
|
pthread_mutex_lock(&gpioScript[n].pthMutex);
|
|
|
|
if (gpioScript[n].run_state == PI_SCRIPT_WAITING)
|
|
{
|
|
gpioScript[n].changedBits =
|
|
gpioScript[n].waitBits & changedBits;
|
|
pthread_cond_signal(&gpioScript[n].pthCond);
|
|
}
|
|
|
|
pthread_mutex_unlock(&gpioScript[n].pthMutex);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (eventBits & scriptEventBits)
|
|
{
|
|
for (n=0; n<PI_MAX_SCRIPTS; n++)
|
|
{
|
|
if ((gpioScript[n].state == PI_SCRIPT_IN_USE) &&
|
|
(gpioScript[n].run_state == PI_SCRIPT_WAITING) &&
|
|
(gpioScript[n].eventBits & eventBits))
|
|
{
|
|
pthread_mutex_lock(&gpioScript[n].pthMutex);
|
|
|
|
if (gpioScript[n].run_state == PI_SCRIPT_WAITING)
|
|
{
|
|
gpioScript[n].changedBits =
|
|
gpioScript[n].eventBits & eventBits;
|
|
pthread_cond_signal(&gpioScript[n].pthCond);
|
|
}
|
|
|
|
pthread_mutex_unlock(&gpioScript[n].pthMutex);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (numSamples) reportedLevel = sample[numSamples-1].level;
|
|
}
|
|
|
|
static void alertWdogCheck(gpioSample_t *sample, int numSamples)
|
|
{
|
|
/*
|
|
Go through and set the last time each GPIO with a watchdog changed state.
|
|
*/
|
|
|
|
int i, j;
|
|
uint32_t LBitV;
|
|
uint32_t bit;
|
|
|
|
for (i=0; i<=PI_MAX_USER_GPIO; i++)
|
|
{
|
|
bit = (1<<i);
|
|
|
|
if (monitorBits & bit & wdogBits)
|
|
{
|
|
LBitV = gpioAlert[i].wdLBitV;
|
|
|
|
for (j=0; j<numSamples; j++)
|
|
{
|
|
if ((sample[j].level & bit) != LBitV)
|
|
{
|
|
LBitV = sample[j].level & bit;
|
|
gpioAlert[i].wdTick = sample[j].tick;
|
|
}
|
|
}
|
|
|
|
gpioAlert[i].wdLBitV = LBitV;
|
|
}
|
|
}
|
|
}
|
|
|
|
static void * pthAlertThread(void *x)
|
|
{
|
|
struct timespec req, rem;
|
|
uint32_t oldLevel, newLevel, level;
|
|
uint32_t oldSlot, newSlot;
|
|
uint32_t expected, ft, sTick;
|
|
uint32_t changedBits;
|
|
int32_t diff, minDiff, stickInited;
|
|
int cycle, pulse;
|
|
int numSamples, ticks, i;
|
|
int rp, reports, totalSamples;
|
|
int stopped;
|
|
int moreToDo;
|
|
gpioSample_t sample[MAX_SAMPLE];
|
|
|
|
req.tv_sec = 0;
|
|
|
|
/* don't start until DMA started */
|
|
|
|
spinWhileStarting();
|
|
|
|
reportedLevel = gpioReg[GPLEV0];
|
|
|
|
oldLevel = reportedLevel;
|
|
|
|
oldSlot = dmaCurrentSlot(dmaNowAtICB());
|
|
|
|
oldSlot = (oldSlot / PULSE_PER_CYCLE) * PULSE_PER_CYCLE;
|
|
|
|
cycle = (oldSlot/PULSE_PER_CYCLE);
|
|
|
|
pulse = 0;
|
|
|
|
stopped = 0;
|
|
|
|
moreToDo = 0;
|
|
|
|
stickInited = 0;
|
|
|
|
sTick = 0;
|
|
|
|
minDiff = gpioCfg.clockMicros / 2;
|
|
|
|
while (1)
|
|
{
|
|
/* Check that DMA is running okay */
|
|
|
|
if (dmaIn[DMA_CONBLK_AD])
|
|
{
|
|
if (stopped)
|
|
{
|
|
DBG(DBG_STARTUP, "****** GOING ******");
|
|
stopped = 0;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
stopped = 1;
|
|
|
|
myGpioDelay(5000);
|
|
|
|
if (runState == PI_RUNNING)
|
|
{
|
|
/* should never be executed, leave code just in case */
|
|
|
|
gpioCfg.internals |= PI_CFG_STATS;
|
|
|
|
dmaInitCbs();
|
|
flushMemory();
|
|
//cast twice to suppress compiler warning, I belive this cast is ok
|
|
//because dmaIbus contains bus addresses, not user addresses. --plugwash
|
|
initDMAgo((uint32_t *)dmaIn, (uint32_t)(uintptr_t)dmaIBus[0]);
|
|
myGpioDelay(5000); /* let DMA run for a while */
|
|
oldSlot = dmaCurrentSlot(dmaNowAtICB());
|
|
gpioStats.DMARestarts++;
|
|
}
|
|
}
|
|
|
|
newSlot = dmaCurrentSlot(dmaNowAtICB());
|
|
|
|
newSlot = (newSlot / PULSE_PER_CYCLE) * PULSE_PER_CYCLE;
|
|
|
|
numSamples = 0;
|
|
|
|
/*
|
|
Extract samples from DMA ring buffer.
|
|
*/
|
|
|
|
while ((oldSlot != newSlot) && (numSamples < MAX_SAMPLE))
|
|
{
|
|
level = myGetLevel(oldSlot++);
|
|
|
|
sample[numSamples].tick = sTick;
|
|
sample[numSamples].level = level;
|
|
|
|
numSamples++;
|
|
|
|
sTick += gpioCfg.clockMicros;
|
|
|
|
if (++pulse >= PULSE_PER_CYCLE)
|
|
{
|
|
pulse = 0;
|
|
|
|
if (++cycle >= bufferCycles)
|
|
{
|
|
cycle = 0;
|
|
oldSlot = 0;
|
|
}
|
|
|
|
expected = sTick;
|
|
|
|
sTick = myGetTick(cycle);
|
|
|
|
if (stickInited)
|
|
{
|
|
diff = sTick - expected;
|
|
|
|
if (abs(diff) > minDiff)
|
|
{
|
|
ft = sample[numSamples-PULSE_PER_CYCLE].tick;
|
|
|
|
ticks = sTick - ft;
|
|
|
|
for (i=1; i<PULSE_PER_CYCLE; i++)
|
|
{
|
|
sample[numSamples-PULSE_PER_CYCLE+i].tick =
|
|
((i*ticks)/PULSE_PER_CYCLE) + ft;
|
|
}
|
|
}
|
|
|
|
diff += (TICKSLOTS/2);
|
|
|
|
if (diff < 0)
|
|
{
|
|
gpioStats.diffTick[0]++;
|
|
}
|
|
|
|
else if (diff >= TICKSLOTS)
|
|
{
|
|
gpioStats.diffTick[TICKSLOTS-1]++;
|
|
}
|
|
|
|
else gpioStats.diffTick[diff]++;
|
|
}
|
|
else
|
|
{
|
|
stickInited = 1;
|
|
numSamples = 0;
|
|
if (!(gpioCfg.ifFlags & PI_DISABLE_ALERT))
|
|
{
|
|
pthAlertRunning = PI_THREAD_RUNNING;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (oldSlot == newSlot) moreToDo = 0; else moreToDo = 1;
|
|
|
|
/* Apply glitch filter */
|
|
|
|
if (numSamples && gFilterBits) alertGlitchFilter(sample, numSamples);
|
|
|
|
/* Apply noise filter */
|
|
|
|
if (numSamples && nFilterBits) alertNoiseFilter(sample, numSamples);
|
|
|
|
/* Compact samples */
|
|
|
|
changedBits = 0;
|
|
oldLevel &= monitorBits;
|
|
reports = 0;
|
|
totalSamples = 0;
|
|
|
|
for (rp=0; rp<numSamples; rp++)
|
|
{
|
|
newLevel = (sample[rp].level & monitorBits);
|
|
|
|
if (newLevel != oldLevel)
|
|
{
|
|
sample[reports].tick = sample[rp].tick;
|
|
sample[reports].level = sample[rp].level;
|
|
changedBits |= (newLevel ^ oldLevel);
|
|
oldLevel = newLevel;
|
|
|
|
reports++;
|
|
|
|
if (reports >= MAX_REPORT)
|
|
{
|
|
totalSamples += reports;
|
|
|
|
/* Rebase watchdog timeouts */
|
|
if (wdogBits) alertWdogCheck(sample, reports);
|
|
|
|
gpioStats.numSamples += reports;
|
|
|
|
alertEmit(sample, reports, changedBits, sample[rp].tick);
|
|
|
|
changedBits = 0;
|
|
reports = 0;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (reports)
|
|
{
|
|
totalSamples += reports;
|
|
|
|
/* Rebase watchdog timeouts */
|
|
if (wdogBits) alertWdogCheck(sample, reports);
|
|
|
|
gpioStats.numSamples += reports;
|
|
}
|
|
|
|
alertEmit(sample, reports, changedBits, sTick);
|
|
reportedLevel = sample[numSamples -1].level;
|
|
|
|
if (totalSamples > gpioStats.maxSamples)
|
|
gpioStats.maxSamples = numSamples;
|
|
|
|
req.tv_sec = 0;
|
|
req.tv_nsec = alert_delays[(gpioCfg.internals>>PI_CFG_ALERT_FREQ)&15];
|
|
|
|
if (moreToDo)
|
|
{
|
|
gpioStats.moreToDo++;
|
|
}
|
|
else
|
|
{
|
|
gpioStats.alertTicks++;
|
|
|
|
while (nanosleep(&req, &rem))
|
|
{
|
|
req.tv_sec = rem.tv_sec;
|
|
req.tv_nsec = rem.tv_nsec;
|
|
}
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/* ======================================================================= */
|
|
|
|
static int scrPop(gpioScript_t *s, int *SP, int *S)
|
|
{
|
|
if ((*SP) > 0)
|
|
{
|
|
return S[--(*SP)];
|
|
}
|
|
else
|
|
{
|
|
s->run_state = PI_SCRIPT_FAILED;
|
|
DBG(DBG_ALWAYS, "script %d too many pops", s->id);
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static void scrPush(gpioScript_t *s, int *SP, int *S, int val)
|
|
{
|
|
if ((*SP) < PI_SCRIPT_STACK_SIZE)
|
|
{
|
|
S[(*SP)++] = val;
|
|
}
|
|
else
|
|
{
|
|
s->run_state = PI_SCRIPT_FAILED;
|
|
DBG(DBG_ALWAYS, "script %d too many pushes", s->id);
|
|
}
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static void scrSwap(int *v1, int *v2)
|
|
{
|
|
int t;
|
|
|
|
t=*v1; *v1=*v2; *v2= t;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static int scrEvtWait(gpioScript_t *s, uint32_t bits)
|
|
{
|
|
pthread_mutex_lock(&s->pthMutex);
|
|
|
|
if (s->request == PI_SCRIPT_RUN)
|
|
{
|
|
s->run_state = PI_SCRIPT_WAITING;
|
|
s->eventBits = bits;
|
|
intScriptEventBits();
|
|
|
|
pthread_cond_wait(&s->pthCond, &s->pthMutex);
|
|
|
|
s->waitBits = 0;
|
|
intScriptEventBits();
|
|
s->run_state = PI_SCRIPT_RUNNING;
|
|
}
|
|
|
|
pthread_mutex_unlock(&s->pthMutex);
|
|
|
|
return s->changedBits;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static int scrWait(gpioScript_t *s, uint32_t bits)
|
|
{
|
|
pthread_mutex_lock(&s->pthMutex);
|
|
|
|
if (s->request == PI_SCRIPT_RUN)
|
|
{
|
|
s->run_state = PI_SCRIPT_WAITING;
|
|
s->waitBits = bits;
|
|
intScriptBits();
|
|
|
|
pthread_cond_wait(&s->pthCond, &s->pthMutex);
|
|
|
|
s->waitBits = 0;
|
|
intScriptBits();
|
|
s->run_state = PI_SCRIPT_RUNNING;
|
|
}
|
|
|
|
pthread_mutex_unlock(&s->pthMutex);
|
|
|
|
return s->changedBits;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static int scrSys(char *cmd, uint32_t p1, uint32_t p2)
|
|
{
|
|
char buf[1024];
|
|
int status;
|
|
|
|
if (!myScriptNameValid(cmd))
|
|
SOFT_ERROR(PI_BAD_SCRIPT_NAME, "bad script name (%s)", cmd);
|
|
|
|
snprintf(buf, sizeof(buf), "/opt/pigpio/cgi/%s %u %u", cmd, p1, p2);
|
|
|
|
DBG(DBG_USER, "%s", buf);
|
|
|
|
status = system(buf);
|
|
|
|
if (status < 0) status = PI_BAD_SHELL_STATUS;
|
|
|
|
return status;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static void *pthScript(void *x)
|
|
{
|
|
gpioScript_t *s;
|
|
cmdInstr_t instr;
|
|
int p1, p2, p1o, p2o, p3o, *t1, *t2;
|
|
int PC, A, F, SP;
|
|
int S[PI_SCRIPT_STACK_SIZE];
|
|
char buf[CMD_MAX_EXTENSION];
|
|
|
|
|
|
S[0] = 0; /* to prevent compiler warning */
|
|
|
|
s = x;
|
|
|
|
while ((volatile int)s->request != PI_SCRIPT_DELETE)
|
|
{
|
|
pthread_mutex_lock(&s->pthMutex);
|
|
s->run_state = PI_SCRIPT_HALTED;
|
|
pthread_cond_wait(&s->pthCond, &s->pthMutex);
|
|
pthread_mutex_unlock(&s->pthMutex);
|
|
|
|
s->run_state = PI_SCRIPT_RUNNING;
|
|
|
|
A = 0;
|
|
F = 0;
|
|
PC = 0;
|
|
SP = 0;
|
|
|
|
while (((volatile int)s->request == PI_SCRIPT_RUN ) &&
|
|
(s->run_state == PI_SCRIPT_RUNNING))
|
|
{
|
|
instr = s->script.instr[PC];
|
|
|
|
p1o = instr.p[1];
|
|
p2o = instr.p[2];
|
|
|
|
if (instr.opt[1] == CMD_VAR) instr.p[1] = s->script.var[p1o];
|
|
else if (instr.opt[1] == CMD_PAR) instr.p[1] = s->script.par[p1o];
|
|
|
|
if (instr.opt[2] == CMD_VAR) instr.p[2] = s->script.var[p2o];
|
|
else if (instr.opt[2] == CMD_PAR) instr.p[2] = s->script.par[p2o];
|
|
/*
|
|
fprintf(stderr, "PC=%d cmd=%d p1o=%d p1=%d p2o=%d p2=%d\n",
|
|
PC, instr.p[0], p1o, instr.p[1], p2o, instr.p[2]);
|
|
fflush(stderr);
|
|
*/
|
|
if (instr.p[0] < PI_CMD_SCRIPT)
|
|
{
|
|
if (instr.p[3])
|
|
{
|
|
if ((instr.p[3] == sizeof(int)) && ((instr.opt[3] == CMD_VAR) || (instr.opt[3] == CMD_PAR)))
|
|
{
|
|
/* Hack to allow register use in 3rd parameter */
|
|
memcpy((char*)&p3o, (char *)instr.p[4], sizeof(int));
|
|
if (instr.opt[3] == CMD_VAR) memcpy(buf, (char *)&(s->script.var[p3o]), sizeof(int));
|
|
else memcpy(buf, (char *)&(s->script.par[p3o]), sizeof(int));
|
|
}
|
|
else
|
|
{
|
|
memcpy(buf, (char *)instr.p[4], instr.p[3]);
|
|
}
|
|
}
|
|
|
|
A = myDoCommand(instr.p, sizeof(buf)-1, buf);
|
|
|
|
F = A;
|
|
|
|
PC++;
|
|
}
|
|
else
|
|
{
|
|
p1 = instr.p[1];
|
|
p2 = instr.p[2];
|
|
|
|
switch (instr.p[0])
|
|
{
|
|
case PI_CMD_ADD: A+=p1; F=A; PC++; break;
|
|
|
|
case PI_CMD_AND: A&=p1; F=A; PC++; break;
|
|
|
|
case PI_CMD_CALL: scrPush(s, &SP, S, PC+1); PC = p1; break;
|
|
|
|
case PI_CMD_CMP: F=A-p1; PC++; break;
|
|
|
|
case PI_CMD_DCR:
|
|
if (instr.opt[1] == CMD_PAR)
|
|
{--s->script.par[p1o]; F=s->script.par[p1o];}
|
|
else
|
|
{--s->script.var[p1o]; F=s->script.var[p1o];}
|
|
PC++;
|
|
break;
|
|
|
|
case PI_CMD_DCRA: --A; F=A; PC++; break;
|
|
|
|
case PI_CMD_DIV: A/=p1; F=A; PC++; break;
|
|
|
|
case PI_CMD_HALT: s->run_state = PI_SCRIPT_HALTED; break;
|
|
|
|
case PI_CMD_EVTWT: A=scrEvtWait(s, p1); F=A; PC++; break;
|
|
|
|
case PI_CMD_INR:
|
|
if (instr.opt[1] == CMD_PAR)
|
|
{++s->script.par[p1o]; F=s->script.par[p1o];}
|
|
else
|
|
{++s->script.var[p1o]; F=s->script.var[p1o];}
|
|
PC++;
|
|
break;
|
|
|
|
case PI_CMD_INRA: ++A; F=A; PC++; break;
|
|
|
|
case PI_CMD_JM: if (F<0) PC=p1; else PC++; break;
|
|
|
|
case PI_CMD_JMP: PC=p1; break;
|
|
|
|
case PI_CMD_JNZ: if (F) PC=p1; else PC++; break;
|
|
|
|
case PI_CMD_JP: if (F>=0) PC=p1; else PC++; break;
|
|
|
|
case PI_CMD_JZ: if (!F) PC=p1; else PC++; break;
|
|
|
|
case PI_CMD_LD:
|
|
if (instr.opt[1] == CMD_PAR) s->script.par[p1o]=p2;
|
|
else s->script.var[p1o]=p2;
|
|
PC++;
|
|
break;
|
|
|
|
case PI_CMD_LDA: A=p1; PC++; break;
|
|
|
|
case PI_CMD_LDAB:
|
|
if ((p1 >= 0) && (p1 < sizeof(buf))) A = buf[p1];
|
|
PC++;
|
|
break;
|
|
|
|
case PI_CMD_MLT: A*=p1; F=A; PC++; break;
|
|
|
|
case PI_CMD_MOD: A%=p1; F=A; PC++; break;
|
|
|
|
case PI_CMD_OR: A|=p1; F=A; PC++; break;
|
|
|
|
case PI_CMD_POP:
|
|
if (instr.opt[1] == CMD_PAR)
|
|
s->script.par[p1o]=scrPop(s, &SP, S);
|
|
else
|
|
s->script.var[p1o]=scrPop(s, &SP, S);
|
|
PC++;
|
|
break;
|
|
|
|
case PI_CMD_POPA: A=scrPop(s, &SP, S); PC++; break;
|
|
|
|
case PI_CMD_PUSH:
|
|
if (instr.opt[1] == CMD_PAR)
|
|
scrPush(s, &SP, S, s->script.par[p1o]);
|
|
else
|
|
scrPush(s, &SP, S, s->script.var[p1o]);
|
|
PC++;
|
|
break;
|
|
|
|
case PI_CMD_PUSHA: scrPush(s, &SP, S, A); PC++; break;
|
|
|
|
case PI_CMD_RET: PC=scrPop(s, &SP, S); break;
|
|
|
|
case PI_CMD_RL:
|
|
if (instr.opt[1] == CMD_PAR)
|
|
{s->script.par[p1o]<<=p2; F=s->script.par[p1o];}
|
|
else
|
|
{s->script.var[p1o]<<=p2; F=s->script.var[p1o];}
|
|
PC++;
|
|
break;
|
|
|
|
case PI_CMD_RLA: A<<=p1; F=A; PC++; break;
|
|
|
|
case PI_CMD_RR:
|
|
if (instr.opt[1] == CMD_PAR)
|
|
{s->script.par[p1o]>>=p2; F=s->script.par[p1o];}
|
|
else
|
|
{s->script.var[p1o]>>=p2; F=s->script.var[p1o];}
|
|
PC++;
|
|
break;
|
|
|
|
case PI_CMD_RRA: A>>=p1; F=A; PC++; break;
|
|
|
|
case PI_CMD_STA:
|
|
if (instr.opt[1] == CMD_PAR) s->script.par[p1o]=A;
|
|
else s->script.var[p1o]=A;
|
|
PC++;
|
|
break;
|
|
|
|
case PI_CMD_STAB:
|
|
if ((p1 >= 0) && (p1 < sizeof(buf))) buf[p1] = A;
|
|
PC++;
|
|
break;
|
|
|
|
case PI_CMD_SUB: A-=p1; F=A; PC++; break;
|
|
|
|
case PI_CMD_SYS:
|
|
A=scrSys((char*)instr.p[4], A, *(gpioReg + GPLEV0));
|
|
F=A;
|
|
PC++;
|
|
break;
|
|
|
|
case PI_CMD_WAIT: A=scrWait(s, p1); F=A; PC++; break;
|
|
|
|
case PI_CMD_X:
|
|
if (instr.opt[1] == CMD_PAR) t1 = &s->script.par[p1o];
|
|
else t1 = &s->script.var[p1o];
|
|
|
|
if (instr.opt[2] == CMD_PAR) t2 = &s->script.par[p2o];
|
|
else t2 = &s->script.var[p2o];
|
|
|
|
scrSwap(t1, t2);
|
|
PC++;
|
|
break;
|
|
|
|
case PI_CMD_XA:
|
|
if (instr.opt[1] == CMD_PAR)
|
|
scrSwap(&s->script.par[p1o], &A);
|
|
else
|
|
scrSwap(&s->script.var[p1o], &A);
|
|
PC++;
|
|
break;
|
|
|
|
case PI_CMD_XOR: A^=p1; F=A; PC++; break;
|
|
|
|
}
|
|
}
|
|
|
|
if (PC >= s->script.instrs) s->run_state = PI_SCRIPT_HALTED;
|
|
|
|
}
|
|
|
|
if ((volatile int)s->request == PI_SCRIPT_HALT)
|
|
s->run_state = PI_SCRIPT_HALTED;
|
|
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static void * pthTimerTick(void *x)
|
|
{
|
|
gpioTimer_t *tp;
|
|
struct timespec req, rem;
|
|
|
|
tp = x;
|
|
|
|
while (1)
|
|
{
|
|
req.tv_sec = tp->millis / THOUSAND;
|
|
req.tv_nsec = (tp->millis % THOUSAND) * THOUSAND * THOUSAND;
|
|
|
|
while (nanosleep(&req, &rem))
|
|
{
|
|
req.tv_sec = rem.tv_sec;
|
|
req.tv_nsec = rem.tv_nsec;
|
|
}
|
|
|
|
if (tp->ex) (tp->func)(tp->userdata);
|
|
else (tp->func)();
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
|
|
static void * pthFifoThread(void *x)
|
|
{
|
|
char buf[CMD_MAX_EXTENSION];
|
|
int idx, flags, len, res, i;
|
|
uintptr_t p[CMD_P_ARR];
|
|
cmdCtlParse_t ctl;
|
|
uint32_t *param;
|
|
char v[CMD_MAX_EXTENSION];
|
|
|
|
myCreatePipe(PI_INPFIFO, 0662);
|
|
|
|
if ((inpFifo = fopen(PI_INPFIFO, "r+")) == NULL)
|
|
SOFT_ERROR((void*)PI_INIT_FAILED, "fopen %s failed(%m)", PI_INPFIFO);
|
|
|
|
myCreatePipe(PI_OUTFIFO, 0664);
|
|
|
|
if ((outFifo = fopen(PI_OUTFIFO, "w+")) == NULL)
|
|
SOFT_ERROR((void*)PI_INIT_FAILED, "fopen %s failed (%m)", PI_OUTFIFO);
|
|
|
|
/* set outFifo non-blocking */
|
|
|
|
flags = fcntl(fileno(outFifo), F_GETFL, 0);
|
|
fcntl(fileno(outFifo), F_SETFL, flags | O_NONBLOCK);
|
|
|
|
/* don't start until DMA started */
|
|
|
|
spinWhileStarting();
|
|
|
|
while (1)
|
|
{
|
|
if (fgets(buf, sizeof(buf), inpFifo) == NULL)
|
|
SOFT_ERROR((void*)PI_INIT_FAILED, "fifo fgets failed (%m)");
|
|
|
|
len = strlen(buf);
|
|
|
|
if (len)
|
|
{
|
|
--len;
|
|
buf[len] = 0; /* replace terminating */
|
|
}
|
|
|
|
ctl.eaten = 0;
|
|
idx = 0;
|
|
|
|
while (((ctl.eaten)<len) && (idx >= 0))
|
|
{
|
|
if ((idx=cmdParse(buf, p, CMD_MAX_EXTENSION, v, &ctl)) >= 0)
|
|
{
|
|
/* make sure extensions are null terminated */
|
|
|
|
v[p[3]] = 0;
|
|
|
|
res = myDoCommand(p, sizeof(v)-1, v);
|
|
|
|
switch (cmdInfo[idx].rv)
|
|
{
|
|
case 0:
|
|
fprintf(outFifo, "%d\n", res);
|
|
break;
|
|
|
|
case 1:
|
|
fprintf(outFifo, "%d\n", res);
|
|
break;
|
|
|
|
case 2:
|
|
fprintf(outFifo, "%d\n", res);
|
|
break;
|
|
|
|
case 3:
|
|
fprintf(outFifo, "%08X\n", res);
|
|
break;
|
|
|
|
case 4:
|
|
fprintf(outFifo, "%u\n", res);
|
|
break;
|
|
|
|
case 5:
|
|
fprintf(outFifo, "%s", cmdUsage);
|
|
break;
|
|
|
|
case 6:
|
|
fprintf(outFifo, "%d", res);
|
|
if (res > 0)
|
|
{
|
|
for (i=0; i<res; i++)
|
|
{
|
|
fprintf(outFifo, " %d", v[i]);
|
|
}
|
|
}
|
|
fprintf(outFifo, "\n");
|
|
break;
|
|
|
|
case 7:
|
|
if (res < 0) fprintf(outFifo, "%d\n", res);
|
|
else
|
|
{
|
|
fprintf(outFifo, "%d", res);
|
|
param = (uint32_t *)v;
|
|
for (i=0; i<PI_MAX_SCRIPT_PARAMS; i++)
|
|
{
|
|
fprintf(outFifo, " %d", param[i]);
|
|
}
|
|
fprintf(outFifo, "\n");
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
else fprintf(outFifo, "%d\n", PI_BAD_FIFO_COMMAND);
|
|
}
|
|
|
|
fflush(outFifo);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static void *pthSocketThreadHandler(void *fdC)
|
|
{
|
|
int sock = *(int*)fdC;
|
|
uintptr_t p[10];
|
|
uint32_t tmp, response[4];
|
|
int i;
|
|
int opt;
|
|
char buf[CMD_MAX_EXTENSION];
|
|
|
|
free(fdC);
|
|
|
|
/* Disable the Nagle algorithm. */
|
|
opt = 1;
|
|
setsockopt(sock, IPPROTO_TCP, TCP_NODELAY, (char*)&opt, sizeof(int));
|
|
|
|
while (1)
|
|
{
|
|
if (sizeof(uintptr_t) == 8)
|
|
{
|
|
if (recv(sock, &tmp, 4, MSG_WAITALL) != 4) break;
|
|
p[0] = (uintptr_t)tmp;
|
|
if (recv(sock, &tmp, 4, MSG_WAITALL) != 4) break;
|
|
p[1] = (uintptr_t)tmp;
|
|
if (recv(sock, &tmp, 4, MSG_WAITALL) != 4) break;
|
|
p[2] = (uintptr_t)tmp;
|
|
if (recv(sock, &tmp, 4, MSG_WAITALL) != 4) break;
|
|
p[3] = (uintptr_t)tmp;
|
|
}
|
|
else
|
|
{
|
|
if (recv(sock, p, 16, MSG_WAITALL) != 16) break;
|
|
}
|
|
|
|
if (p[3])
|
|
{
|
|
if (p[3] < sizeof(buf))
|
|
{
|
|
/* read extension into buf */
|
|
if (recv(sock, buf, p[3], MSG_WAITALL) != p[3])
|
|
{
|
|
/* Serious error. No point continuing. */
|
|
DBG(DBG_ALWAYS,
|
|
"recv failed for %"PRIdPTR" bytes, sock=%d", p[3], sock);
|
|
|
|
closeOrphanedNotifications(-1, sock);
|
|
|
|
close(sock);
|
|
|
|
return 0;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
/* Serious error. No point continuing. */
|
|
DBG(DBG_ALWAYS, "ext too large %"PRIdPTR"(%zd), sock=%d",
|
|
p[3], sizeof(buf), sock);
|
|
|
|
closeOrphanedNotifications(-1, sock);
|
|
|
|
close(sock);
|
|
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
/* add null terminator in case it's a string */
|
|
|
|
buf[p[3]] = 0;
|
|
|
|
switch (p[0])
|
|
{
|
|
case PI_CMD_NOIB:
|
|
|
|
p[3] = gpioNotifyOpenInBand(sock);
|
|
|
|
/* Enable the Nagle algorithm. */
|
|
opt = 0;
|
|
setsockopt(
|
|
sock, IPPROTO_TCP, TCP_NODELAY, (char*)&opt, sizeof(int));
|
|
|
|
break;
|
|
|
|
case PI_CMD_PROCP:
|
|
p[3] = myDoCommand(p, sizeof(buf)-1, buf+sizeof(int));
|
|
if (((int)p[3]) >= 0)
|
|
{
|
|
memcpy(buf, &p[3], 4);
|
|
p[3] = 4 + (4*PI_MAX_SCRIPT_PARAMS);
|
|
}
|
|
break;
|
|
|
|
default:
|
|
p[3] = myDoCommand(p, sizeof(buf)-1, buf);
|
|
}
|
|
|
|
if (sizeof(uintptr_t) == 8) // 64-bit system
|
|
{
|
|
for (i = 0; i < 4; i++)
|
|
response[i] = (uint32_t)p[i];
|
|
if (write(sock, response, 16) == -1) { /* ignore errors */ }
|
|
}
|
|
else // 32-bit system
|
|
{
|
|
if (write(sock, p, 16) == -1) { /* ignore errors */ }
|
|
}
|
|
|
|
switch (p[0])
|
|
{
|
|
/* extensions */
|
|
|
|
case PI_CMD_BI2CZ:
|
|
case PI_CMD_BSCX:
|
|
case PI_CMD_CF2:
|
|
case PI_CMD_FL:
|
|
case PI_CMD_FR:
|
|
case PI_CMD_I2CPK:
|
|
case PI_CMD_I2CRD:
|
|
case PI_CMD_I2CRI:
|
|
case PI_CMD_I2CRK:
|
|
case PI_CMD_I2CZ:
|
|
case PI_CMD_PROCP:
|
|
case PI_CMD_SERR:
|
|
case PI_CMD_SLR:
|
|
case PI_CMD_SPIX:
|
|
case PI_CMD_SPIR:
|
|
case PI_CMD_BSPIX:
|
|
|
|
if (((int)p[3]) > 0)
|
|
{
|
|
if (write(sock, buf, p[3]) == 1) { /* ignore errors */ }
|
|
}
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
closeOrphanedNotifications(-1, sock);
|
|
|
|
close(sock);
|
|
|
|
DBG(DBG_USER, "Socket %d closed", sock);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int addrAllowed(struct sockaddr *saddr)
|
|
{
|
|
int i;
|
|
uint32_t addr;
|
|
|
|
if (!numSockNetAddr) return 1;
|
|
|
|
// FIXME: add IPv6 whitelisting support
|
|
if (saddr->sa_family != AF_INET) return 0;
|
|
|
|
addr = ((struct sockaddr_in *) saddr)->sin_addr.s_addr;
|
|
|
|
for (i=0; i<numSockNetAddr; i++)
|
|
{
|
|
if (addr == sockNetAddr[i]) return 1;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static void * pthSocketThread(void *x)
|
|
{
|
|
int fdC=0, c, *sock;
|
|
struct sockaddr_storage client;
|
|
pthread_attr_t attr;
|
|
|
|
if (pthread_attr_init(&attr))
|
|
SOFT_ERROR((void*)PI_INIT_FAILED,
|
|
"pthread_attr_init failed (%m)");
|
|
|
|
if (pthread_attr_setstacksize(&attr, STACK_SIZE))
|
|
SOFT_ERROR((void*)PI_INIT_FAILED,
|
|
"pthread_attr_setstacksize failed (%m)");
|
|
|
|
if (pthread_attr_setdetachstate(&attr, PTHREAD_CREATE_DETACHED))
|
|
SOFT_ERROR((void*)PI_INIT_FAILED,
|
|
"pthread_attr_setdetachstate failed (%m)");
|
|
|
|
/* fdSock opened in gpioInitialise so that we can treat
|
|
failure to bind as fatal. */
|
|
|
|
listen(fdSock, 100);
|
|
|
|
c = sizeof(client);
|
|
|
|
/* don't start until DMA started */
|
|
|
|
spinWhileStarting();
|
|
|
|
while (fdC >= 0)
|
|
{
|
|
pthread_t thr;
|
|
|
|
fdC = accept(fdSock, (struct sockaddr *)&client, (socklen_t*)&c);
|
|
|
|
closeOrphanedNotifications(-1, fdC);
|
|
|
|
if (addrAllowed((struct sockaddr *)&client))
|
|
{
|
|
DBG(DBG_USER, "Connection accepted on socket %d", fdC);
|
|
|
|
sock = malloc(sizeof(int));
|
|
|
|
*sock = fdC;
|
|
|
|
/* Enable tcp_keepalive */
|
|
int optval = 1;
|
|
socklen_t optlen = sizeof(optval);
|
|
|
|
if (setsockopt(fdC, SOL_SOCKET, SO_KEEPALIVE, &optval, optlen) < 0)
|
|
{
|
|
DBG(DBG_ALWAYS, "setsockopt() fail, closing socket %d", fdC);
|
|
close(fdC);
|
|
}
|
|
|
|
DBG(DBG_USER, "SO_KEEPALIVE enabled on socket %d\n", fdC);
|
|
|
|
if (pthread_create
|
|
(&thr, &attr, pthSocketThreadHandler, (void*) sock) < 0)
|
|
SOFT_ERROR((void*)PI_INIT_FAILED,
|
|
"socket pthread_create failed (%m)");
|
|
}
|
|
else
|
|
{
|
|
DBG(DBG_ALWAYS, "Connection rejected, closing");
|
|
close(fdC);
|
|
}
|
|
}
|
|
|
|
if (fdC < 0)
|
|
SOFT_ERROR((void*)PI_INIT_FAILED, "accept failed (%m)");
|
|
|
|
return 0;
|
|
}
|
|
|
|
/* ======================================================================= */
|
|
|
|
static void initCheckLockFile(void)
|
|
{
|
|
int fd;
|
|
int count;
|
|
int pid;
|
|
int err;
|
|
int delete;
|
|
char str[20];
|
|
|
|
fd = open(PI_LOCKFILE, O_RDONLY);
|
|
if(errno == ENOENT)
|
|
errno = 0; // reset if expected error occured
|
|
|
|
if (fd != -1)
|
|
{
|
|
DBG(DBG_STARTUP, "lock file exists");
|
|
delete = 1;
|
|
|
|
count = read(fd, str, sizeof(str)-1);
|
|
|
|
if (count)
|
|
{
|
|
pid = atoi(str);
|
|
err = kill(pid, 0);
|
|
if (!err) delete = 0; /* process still exists */
|
|
DBG(DBG_STARTUP, "lock file pid=%d err=%d", pid, err);
|
|
}
|
|
|
|
close(fd);
|
|
DBG(DBG_STARTUP, "lock file delete=%d", delete);
|
|
|
|
if (delete) unlink(PI_LOCKFILE);
|
|
}
|
|
}
|
|
|
|
static int initGrabLockFile(void)
|
|
{
|
|
int fd;
|
|
int lockResult;
|
|
char pidStr[20];
|
|
|
|
initCheckLockFile();
|
|
|
|
/* try to grab the lock file */
|
|
|
|
fd = open(PI_LOCKFILE, O_WRONLY|O_CREAT|O_EXCL|O_TRUNC, 0644);
|
|
|
|
if (fd != -1)
|
|
{
|
|
lockResult = flock(fd, LOCK_EX|LOCK_NB);
|
|
|
|
if(lockResult == 0)
|
|
{
|
|
sprintf(pidStr, "%d\n", (int)getpid());
|
|
|
|
if (write(fd, pidStr, strlen(pidStr)) == -1)
|
|
{
|
|
/* ignore errors */
|
|
}
|
|
}
|
|
else
|
|
{
|
|
close(fd);
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
return fd;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static uint32_t * initMapMem(int fd, uint32_t addr, uint32_t len)
|
|
{
|
|
return (uint32_t *) mmap(0, len,
|
|
PROT_READ|PROT_WRITE,
|
|
MAP_SHARED|MAP_LOCKED,
|
|
fd, addr);
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static int initCheckPermitted(void)
|
|
{
|
|
DBG(DBG_STARTUP, "");
|
|
|
|
if (!pi_ispi)
|
|
{
|
|
DBG(DBG_ALWAYS,
|
|
"\n" \
|
|
"+---------------------------------------------------------+\n" \
|
|
"|Sorry, this system does not appear to be a raspberry pi. |\n" \
|
|
"|aborting. |\n" \
|
|
"+---------------------------------------------------------+\n\n");
|
|
return -1;
|
|
}
|
|
|
|
if ((fdMem = open("/dev/mem", O_RDWR | O_SYNC) ) < 0)
|
|
{
|
|
DBG(DBG_ALWAYS,
|
|
"\n" \
|
|
"+---------------------------------------------------------+\n" \
|
|
"|Sorry, you don't have permission to run this program. |\n" \
|
|
"|Try running as root, e.g. precede the command with sudo. |\n" \
|
|
"+---------------------------------------------------------+\n\n");
|
|
return -1;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static int initPeripherals(void)
|
|
{
|
|
DBG(DBG_STARTUP, "");
|
|
|
|
gpioReg = initMapMem(fdMem, GPIO_BASE, GPIO_LEN);
|
|
|
|
if (gpioReg == MAP_FAILED)
|
|
SOFT_ERROR(PI_INIT_FAILED, "mmap gpio failed (%m)");
|
|
|
|
dmaReg = initMapMem(fdMem, DMA_BASE, DMA_LEN);
|
|
|
|
if (dmaReg == MAP_FAILED)
|
|
SOFT_ERROR(PI_INIT_FAILED, "mmap dma failed (%m)");
|
|
|
|
/* we should know if we are running on a BCM2711 by now */
|
|
|
|
if (gpioCfg.DMAprimaryChannel == PI_DEFAULT_DMA_NOT_SET)
|
|
{
|
|
if (pi_is_2711)
|
|
gpioCfg.DMAprimaryChannel = PI_DEFAULT_DMA_PRIMARY_CH_2711;
|
|
else
|
|
gpioCfg.DMAprimaryChannel = PI_DEFAULT_DMA_PRIMARY_CHANNEL;
|
|
}
|
|
|
|
if (gpioCfg.DMAsecondaryChannel == PI_DEFAULT_DMA_NOT_SET)
|
|
{
|
|
if (pi_is_2711)
|
|
gpioCfg.DMAsecondaryChannel = PI_DEFAULT_DMA_SECONDARY_CH_2711;
|
|
else
|
|
gpioCfg.DMAsecondaryChannel = PI_DEFAULT_DMA_SECONDARY_CHANNEL;
|
|
}
|
|
|
|
dmaIn = dmaReg + (gpioCfg.DMAprimaryChannel * 0x40);
|
|
dmaOut = dmaReg + (gpioCfg.DMAsecondaryChannel * 0x40);
|
|
|
|
DBG(DBG_STARTUP, "DMA #%d @ %08"PRIXPTR,
|
|
gpioCfg.DMAprimaryChannel, (uintptr_t)dmaIn);
|
|
|
|
DBG(DBG_STARTUP, "debug reg is %08X", dmaIn[DMA_DEBUG]);
|
|
|
|
clkReg = initMapMem(fdMem, CLK_BASE, CLK_LEN);
|
|
|
|
if (clkReg == MAP_FAILED)
|
|
SOFT_ERROR(PI_INIT_FAILED, "mmap clk failed (%m)");
|
|
|
|
systReg = initMapMem(fdMem, SYST_BASE, SYST_LEN);
|
|
|
|
if (systReg == MAP_FAILED)
|
|
SOFT_ERROR(PI_INIT_FAILED, "mmap syst failed (%m)");
|
|
|
|
spiReg = initMapMem(fdMem, SPI_BASE, SPI_LEN);
|
|
|
|
if (spiReg == MAP_FAILED)
|
|
SOFT_ERROR(PI_INIT_FAILED, "mmap spi failed (%m)");
|
|
|
|
pwmReg = initMapMem(fdMem, PWM_BASE, PWM_LEN);
|
|
|
|
if (pwmReg == MAP_FAILED)
|
|
SOFT_ERROR(PI_INIT_FAILED, "mmap pwm failed (%m)");
|
|
|
|
pcmReg = initMapMem(fdMem, PCM_BASE, PCM_LEN);
|
|
|
|
if (pcmReg == MAP_FAILED)
|
|
SOFT_ERROR(PI_INIT_FAILED, "mmap pcm failed (%m)");
|
|
|
|
auxReg = initMapMem(fdMem, AUX_BASE, AUX_LEN);
|
|
|
|
if (auxReg == MAP_FAILED)
|
|
SOFT_ERROR(PI_INIT_FAILED, "mmap aux failed (%m)");
|
|
|
|
padsReg = initMapMem(fdMem, PADS_BASE, PADS_LEN);
|
|
|
|
if (padsReg == MAP_FAILED)
|
|
SOFT_ERROR(PI_INIT_FAILED, "mmap pads failed (%m)");
|
|
|
|
bscsReg = initMapMem(fdMem, BSCS_BASE, BSCS_LEN);
|
|
|
|
if (bscsReg == MAP_FAILED)
|
|
SOFT_ERROR(PI_INIT_FAILED, "mmap bscs failed (%m)");
|
|
|
|
return 0;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static int initZaps
|
|
(int pmapFd, void *virtualBase, int basePage, int pages)
|
|
{
|
|
int n;
|
|
uintptr_t index;
|
|
off_t offset;
|
|
ssize_t t;
|
|
uint32_t physical;
|
|
int status;
|
|
uintptr_t pageAdr;
|
|
unsigned long long pa;
|
|
|
|
DBG(DBG_STARTUP, "");
|
|
|
|
status = 0;
|
|
|
|
pageAdr = (uintptr_t) dmaVirt[basePage];
|
|
|
|
index = ((uintptr_t)virtualBase / PAGE_SIZE) * 8;
|
|
|
|
offset = lseek(pmapFd, index, SEEK_SET);
|
|
|
|
if (offset != index)
|
|
SOFT_ERROR(PI_INIT_FAILED, "lseek pagemap failed (%m)");
|
|
|
|
for (n=0; n<pages; n++)
|
|
{
|
|
t = read(pmapFd, &pa, sizeof(pa));
|
|
|
|
if (t != sizeof(pa))
|
|
SOFT_ERROR(PI_INIT_FAILED, "read pagemap failed (%m)");
|
|
|
|
DBG(DBG_STARTUP, "pf%d=%016llX", n, pa);
|
|
|
|
physical = 0x3FFFFFFF & (PAGE_SIZE * (pa & 0xFFFFFFFF));
|
|
|
|
if (physical)
|
|
{
|
|
//cast twice to suppress warning, I belive this is ok as these
|
|
//are bus addresses, not virtual addresses. --plugwash
|
|
dmaBus[basePage+n] = (dmaPage_t *)(uintptr_t) (physical | pi_dram_bus);
|
|
|
|
dmaVirt[basePage+n] = mmap
|
|
(
|
|
(void *)pageAdr,
|
|
PAGE_SIZE,
|
|
PROT_READ|PROT_WRITE,
|
|
MAP_SHARED|MAP_FIXED|MAP_LOCKED|MAP_NORESERVE,
|
|
fdMem,
|
|
physical
|
|
);
|
|
}
|
|
else status = 1;
|
|
|
|
pageAdr += PAGE_SIZE;
|
|
}
|
|
|
|
return status;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static int initPagemapBlock(int block)
|
|
{
|
|
int trys, ok;
|
|
unsigned pageNum;
|
|
|
|
DBG(DBG_STARTUP, "block=%d", block);
|
|
|
|
dmaPMapBlk[block] = mmap(
|
|
0, (PAGES_PER_BLOCK*PAGE_SIZE),
|
|
PROT_READ|PROT_WRITE,
|
|
MAP_SHARED|MAP_ANONYMOUS|MAP_NORESERVE|MAP_LOCKED,
|
|
-1, 0);
|
|
|
|
if (dmaPMapBlk[block] == MAP_FAILED)
|
|
SOFT_ERROR(PI_INIT_FAILED, "mmap dma block %d failed (%m)", block);
|
|
|
|
/* force allocation of physical memory */
|
|
|
|
memset((void *)dmaPMapBlk[block], 0xAA, (PAGES_PER_BLOCK*PAGE_SIZE));
|
|
|
|
memset((void *)dmaPMapBlk[block], 0xFF, (PAGES_PER_BLOCK*PAGE_SIZE));
|
|
|
|
memset((void *)dmaPMapBlk[block], 0, (PAGES_PER_BLOCK*PAGE_SIZE));
|
|
|
|
pageNum = block * PAGES_PER_BLOCK;
|
|
|
|
dmaVirt[pageNum] = mmap(
|
|
0, (PAGES_PER_BLOCK*PAGE_SIZE),
|
|
PROT_READ|PROT_WRITE,
|
|
MAP_SHARED|MAP_ANONYMOUS|MAP_NORESERVE|MAP_LOCKED,
|
|
-1, 0);
|
|
|
|
if (dmaVirt[pageNum] == MAP_FAILED)
|
|
SOFT_ERROR(PI_INIT_FAILED, "mmap dma block %d failed (%m)", block);
|
|
|
|
munmap(dmaVirt[pageNum], PAGES_PER_BLOCK*PAGE_SIZE);
|
|
|
|
trys = 0;
|
|
ok = 0;
|
|
|
|
while ((trys < 10) && !ok)
|
|
{
|
|
if (initZaps(fdPmap,
|
|
dmaPMapBlk[block],
|
|
pageNum,
|
|
PAGES_PER_BLOCK) == 0) ok = 1;
|
|
else myGpioDelay(50000);
|
|
|
|
++trys;
|
|
}
|
|
|
|
if (!ok) SOFT_ERROR(PI_INIT_FAILED, "initZaps failed");
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int initMboxBlock(int block)
|
|
{
|
|
int n, ok;
|
|
unsigned page;
|
|
uintptr_t virtualAdr;
|
|
uintptr_t busAdr;
|
|
|
|
DBG(DBG_STARTUP, "block=%d", block);
|
|
|
|
ok = mbDMAAlloc
|
|
(&dmaMboxBlk[block], PAGES_PER_BLOCK * PAGE_SIZE, pi_mem_flag);
|
|
|
|
if (!ok) SOFT_ERROR(PI_INIT_FAILED, "init mbox zaps failed");
|
|
|
|
page = block * PAGES_PER_BLOCK;
|
|
|
|
virtualAdr = (uintptr_t) dmaMboxBlk[block].virtual_addr;
|
|
busAdr = dmaMboxBlk[block].bus_addr;
|
|
|
|
for (n=0; n<PAGES_PER_BLOCK; n++)
|
|
{
|
|
dmaVirt[page+n] = (dmaPage_t *) virtualAdr;
|
|
dmaBus[page+n] = (dmaPage_t *) busAdr;
|
|
virtualAdr += PAGE_SIZE;
|
|
busAdr += PAGE_SIZE;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static int initAllocDMAMem(void)
|
|
{
|
|
int i, servoCycles, superCycles;
|
|
int status;
|
|
|
|
DBG(DBG_STARTUP, "");
|
|
|
|
/* Calculate the number of blocks needed for buffers. The number
|
|
of blocks must be a multiple of the 20ms servo cycle.
|
|
*/
|
|
|
|
servoCycles = gpioCfg.bufferMilliseconds / 20;
|
|
if (gpioCfg.bufferMilliseconds % 20) servoCycles++;
|
|
|
|
bufferCycles = (SUPERCYCLE * servoCycles) / gpioCfg.clockMicros;
|
|
|
|
superCycles = bufferCycles / SUPERCYCLE;
|
|
if (bufferCycles % SUPERCYCLE) superCycles++;
|
|
|
|
bufferCycles = SUPERCYCLE * superCycles;
|
|
|
|
bufferBlocks = bufferCycles / CYCLES_PER_BLOCK;
|
|
|
|
DBG(DBG_STARTUP, "bmillis=%d mics=%d bblk=%d bcyc=%d",
|
|
gpioCfg.bufferMilliseconds, gpioCfg.clockMicros,
|
|
bufferBlocks, bufferCycles);
|
|
|
|
/* allocate memory for pointers to virtual and bus memory pages */
|
|
|
|
dmaVirt = mmap(
|
|
0, PAGES_PER_BLOCK*(bufferBlocks+PI_WAVE_BLOCKS)*sizeof(dmaPage_t *),
|
|
PROT_READ|PROT_WRITE,
|
|
MAP_PRIVATE|MAP_ANONYMOUS|MAP_LOCKED,
|
|
-1, 0);
|
|
|
|
if (dmaVirt == MAP_FAILED)
|
|
SOFT_ERROR(PI_INIT_FAILED, "mmap dma virtual failed (%m)");
|
|
|
|
dmaBus = mmap(
|
|
0, PAGES_PER_BLOCK*(bufferBlocks+PI_WAVE_BLOCKS)*sizeof(dmaPage_t *),
|
|
PROT_READ|PROT_WRITE,
|
|
MAP_PRIVATE|MAP_ANONYMOUS|MAP_LOCKED,
|
|
-1, 0);
|
|
|
|
if (dmaBus == MAP_FAILED)
|
|
SOFT_ERROR(PI_INIT_FAILED, "mmap dma bus failed (%m)");
|
|
|
|
dmaIVirt = (dmaIPage_t **) dmaVirt;
|
|
dmaIBus = (dmaIPage_t **) dmaBus;
|
|
|
|
dmaOVirt = (dmaOPage_t **)(dmaVirt + (PAGES_PER_BLOCK*bufferBlocks));
|
|
dmaOBus = (dmaOPage_t **)(dmaBus + (PAGES_PER_BLOCK*bufferBlocks));
|
|
|
|
if ((gpioCfg.memAllocMode == PI_MEM_ALLOC_PAGEMAP) ||
|
|
((gpioCfg.memAllocMode == PI_MEM_ALLOC_AUTO) &&
|
|
(gpioCfg.bufferMilliseconds > PI_DEFAULT_BUFFER_MILLIS)))
|
|
{
|
|
/* pagemap allocation of DMA memory */
|
|
|
|
dmaPMapBlk = mmap(
|
|
0, (bufferBlocks+PI_WAVE_BLOCKS)*sizeof(dmaPage_t *),
|
|
PROT_READ|PROT_WRITE,
|
|
MAP_PRIVATE|MAP_ANONYMOUS|MAP_LOCKED,
|
|
-1, 0);
|
|
|
|
if (dmaPMapBlk == MAP_FAILED)
|
|
SOFT_ERROR(PI_INIT_FAILED, "pagemap mmap block failed (%m)");
|
|
|
|
fdPmap = open("/proc/self/pagemap", O_RDONLY);
|
|
|
|
if (fdPmap < 0)
|
|
SOFT_ERROR(PI_INIT_FAILED, "pagemap open failed(%m)");
|
|
|
|
for (i=0; i<(bufferBlocks+PI_WAVE_BLOCKS); i++)
|
|
{
|
|
status = initPagemapBlock(i);
|
|
if (status < 0)
|
|
{
|
|
close(fdPmap);
|
|
return status;
|
|
}
|
|
}
|
|
|
|
close(fdPmap);
|
|
|
|
DBG(DBG_STARTUP, "dmaPMapBlk=%08"PRIXPTR" dmaIn=%08"PRIXPTR,
|
|
(uintptr_t)dmaPMapBlk, (uintptr_t)dmaIn);
|
|
}
|
|
else
|
|
{
|
|
/* mailbox allocation of DMA memory */
|
|
|
|
dmaMboxBlk = mmap(
|
|
0, (bufferBlocks+PI_WAVE_BLOCKS)*sizeof(DMAMem_t),
|
|
PROT_READ|PROT_WRITE,
|
|
MAP_PRIVATE|MAP_ANONYMOUS|MAP_LOCKED,
|
|
-1, 0);
|
|
|
|
if (dmaMboxBlk == MAP_FAILED)
|
|
SOFT_ERROR(PI_INIT_FAILED, "mmap mbox block failed (%m)");
|
|
|
|
fdMbox = mbOpen();
|
|
|
|
if (fdMbox < 0)
|
|
SOFT_ERROR(PI_INIT_FAILED, "mbox open failed(%m)");
|
|
|
|
for (i=0; i<(bufferBlocks+PI_WAVE_BLOCKS); i++)
|
|
{
|
|
status = initMboxBlock(i);
|
|
if (status < 0)
|
|
{
|
|
mbClose(fdMbox);
|
|
return status;
|
|
}
|
|
}
|
|
|
|
mbClose(fdMbox);
|
|
|
|
DBG(DBG_STARTUP, "dmaMboxBlk=%08"PRIXPTR" dmaIn=%08"PRIXPTR,
|
|
(uintptr_t)dmaMboxBlk, (uintptr_t)dmaIn);
|
|
}
|
|
|
|
DBG(DBG_STARTUP,
|
|
"gpioReg=%08"PRIXPTR" pwmReg=%08"PRIXPTR" pcmReg=%08"PRIXPTR" clkReg=%08"PRIXPTR" auxReg=%08"PRIXPTR,
|
|
(uintptr_t)gpioReg, (uintptr_t)pwmReg,
|
|
(uintptr_t)pcmReg, (uintptr_t)clkReg, (uintptr_t)auxReg);
|
|
|
|
for (i=0; i<DMAI_PAGES; i++)
|
|
DBG(DBG_STARTUP, "dmaIBus[%d]=%08"PRIXPTR, i, (uintptr_t)dmaIBus[i]);
|
|
|
|
if (gpioCfg.dbgLevel >= DBG_DMACBS)
|
|
{
|
|
fprintf(stderr, "*** INPUT DMA CONTROL BLOCKS ***\n");
|
|
for (i=0; i<NUM_CBS; i++) dmaCbPrint(i);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static void initPWM(unsigned bits)
|
|
{
|
|
DBG(DBG_STARTUP, "bits=%d", bits);
|
|
|
|
/* reset PWM */
|
|
|
|
pwmReg[PWM_CTL] = 0;
|
|
|
|
myGpioDelay(10);
|
|
|
|
pwmReg[PWM_STA] = -1;
|
|
|
|
myGpioDelay(10);
|
|
|
|
/* set number of bits to transmit */
|
|
|
|
pwmReg[PWM_RNG1] = bits;
|
|
|
|
myGpioDelay(10);
|
|
|
|
dmaIVirt[0]->periphData = 1;
|
|
|
|
/* enable PWM DMA, raise panic and dreq thresholds to 15 */
|
|
|
|
pwmReg[PWM_DMAC] = PWM_DMAC_ENAB |
|
|
PWM_DMAC_PANIC(15) |
|
|
PWM_DMAC_DREQ(15);
|
|
|
|
myGpioDelay(10);
|
|
|
|
/* clear PWM fifo */
|
|
|
|
pwmReg[PWM_CTL] = PWM_CTL_CLRF1;
|
|
|
|
myGpioDelay(10);
|
|
|
|
/* enable PWM channel 1 and use fifo */
|
|
|
|
pwmReg[PWM_CTL] = PWM_CTL_USEF1 | PWM_CTL_MODE1 | PWM_CTL_PWEN1;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static void initPCM(unsigned bits)
|
|
{
|
|
DBG(DBG_STARTUP, "bits=%d", bits);
|
|
|
|
/* disable PCM so we can modify the regs */
|
|
|
|
pcmReg[PCM_CS] = 0;
|
|
|
|
myGpioDelay(1000);
|
|
|
|
pcmReg[PCM_FIFO] = 0;
|
|
pcmReg[PCM_MODE] = 0;
|
|
pcmReg[PCM_RXC] = 0;
|
|
pcmReg[PCM_TXC] = 0;
|
|
pcmReg[PCM_DREQ] = 0;
|
|
pcmReg[PCM_INTEN] = 0;
|
|
pcmReg[PCM_INTSTC] = 0;
|
|
pcmReg[PCM_GRAY] = 0;
|
|
|
|
myGpioDelay(1000);
|
|
|
|
pcmReg[PCM_MODE] = PCM_MODE_FLEN(bits-1); /* # bits in frame */
|
|
|
|
/* enable channel 1 with # bits width */
|
|
|
|
pcmReg[PCM_TXC] = PCM_TXC_CH1EN | PCM_TXC_CH1WID(bits-8);
|
|
|
|
pcmReg[PCM_CS] |= PCM_CS_STBY; /* clear standby */
|
|
|
|
myGpioDelay(1000);
|
|
|
|
pcmReg[PCM_CS] |= PCM_CS_TXCLR; /* clear TX FIFO */
|
|
|
|
pcmReg[PCM_CS] |= PCM_CS_DMAEN; /* enable DREQ */
|
|
|
|
pcmReg[PCM_DREQ] = PCM_DREQ_TX_PANIC(16) | PCM_DREQ_TX_REQ_L(30);
|
|
|
|
pcmReg[PCM_INTSTC] = 0b1111; /* clear status bits */
|
|
|
|
/* enable PCM */
|
|
|
|
pcmReg[PCM_CS] |= PCM_CS_EN;
|
|
|
|
/* enable tx */
|
|
|
|
pcmReg[PCM_CS] |= PCM_CS_TXON;
|
|
|
|
dmaIVirt[0]->periphData = 0x0F;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static void initHWClk
|
|
(int clkCtl, int clkDiv, int clkSrc, int divI, int divF, int MASH)
|
|
{
|
|
DBG(DBG_INTERNAL, "ctl=%d div=%d src=%d /I=%d /f=%d M=%d",
|
|
clkCtl, clkDiv, clkSrc, divI, divF, MASH);
|
|
|
|
/* kill the clock if busy, anything else isn't reliable */
|
|
|
|
if (clkReg[clkCtl] & CLK_CTL_BUSY)
|
|
{
|
|
do
|
|
{
|
|
clkReg[clkCtl] = BCM_PASSWD | CLK_CTL_KILL;
|
|
}
|
|
while (clkReg[clkCtl] & CLK_CTL_BUSY);
|
|
}
|
|
|
|
clkReg[clkDiv] = (BCM_PASSWD | CLK_DIV_DIVI(divI) | CLK_DIV_DIVF(divF));
|
|
|
|
usleep(10);
|
|
|
|
clkReg[clkCtl] = (BCM_PASSWD | CLK_CTL_MASH(MASH) | CLK_CTL_SRC(clkSrc));
|
|
|
|
usleep(10);
|
|
|
|
clkReg[clkCtl] |= (BCM_PASSWD | CLK_CTL_ENAB);
|
|
}
|
|
|
|
static void initClock(int mainClock)
|
|
{
|
|
const unsigned BITS=10;
|
|
int clockPWM;
|
|
unsigned clkCtl, clkDiv, clkSrc, clkDivI, clkDivF, clkMash, clkBits;
|
|
char *per;
|
|
unsigned micros;
|
|
|
|
DBG(DBG_STARTUP, "mainClock=%d", mainClock);
|
|
|
|
if (mainClock) micros = gpioCfg.clockMicros;
|
|
else micros = PI_WF_MICROS;
|
|
|
|
clockPWM = mainClock ^ (gpioCfg.clockPeriph == PI_CLOCK_PCM);
|
|
|
|
if (clockPWM)
|
|
{
|
|
clkCtl = CLK_PWMCTL;
|
|
clkDiv = CLK_PWMDIV;
|
|
per = "PWM";
|
|
}
|
|
else
|
|
{
|
|
clkCtl = CLK_PCMCTL;
|
|
clkDiv = CLK_PCMDIV;
|
|
per = "PCM";
|
|
}
|
|
|
|
clkSrc = CLK_CTL_SRC_PLLD;
|
|
clkDivI = clk_plld_freq / (10000000 / micros); /* 10 MHz - 1 MHz */
|
|
clkBits = BITS; /* 10/BITS MHz - 1/BITS MHz */
|
|
clkDivF = 0;
|
|
clkMash = 0;
|
|
|
|
DBG(DBG_STARTUP, "%s PLLD divi=%d divf=%d mash=%d bits=%d",
|
|
per, clkDivI, clkDivF, clkMash, clkBits);
|
|
|
|
initHWClk(clkCtl, clkDiv, clkSrc, clkDivI, clkDivF, clkMash);
|
|
|
|
if (clockPWM) initPWM(BITS);
|
|
else initPCM(BITS);
|
|
|
|
myGpioDelay(2000);
|
|
}
|
|
|
|
static void initKillDMA(volatile uint32_t *dmaAddr)
|
|
{
|
|
dmaAddr[DMA_CS] = DMA_CHANNEL_ABORT;
|
|
dmaAddr[DMA_CS] = 0;
|
|
dmaAddr[DMA_CS] = DMA_CHANNEL_RESET;
|
|
|
|
dmaAddr[DMA_CONBLK_AD] = 0;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static void initDMAgo(volatile uint32_t *dmaAddr, uint32_t cbAddr)
|
|
{
|
|
DBG(DBG_STARTUP, "");
|
|
|
|
initKillDMA(dmaAddr);
|
|
|
|
dmaAddr[DMA_CS] = DMA_INTERRUPT_STATUS | DMA_END_FLAG;
|
|
|
|
dmaAddr[DMA_CONBLK_AD] = cbAddr;
|
|
|
|
/* clear READ/FIFO/READ_LAST_NOT_SET error bits */
|
|
|
|
dmaAddr[DMA_DEBUG] = DMA_DEBUG_READ_ERR |
|
|
DMA_DEBUG_FIFO_ERR |
|
|
DMA_DEBUG_RD_LST_NOT_SET_ERR;
|
|
|
|
|
|
dmaAddr[DMA_CS] = DMA_WAIT_ON_WRITES |
|
|
DMA_PANIC_PRIORITY(8) |
|
|
DMA_PRIORITY(8) |
|
|
DMA_ACTIVE;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static void initClearGlobals(void)
|
|
{
|
|
int i;
|
|
|
|
DBG(DBG_STARTUP, "");
|
|
|
|
alertBits = 0;
|
|
monitorBits = 0;
|
|
notifyBits = 0;
|
|
scriptBits = 0;
|
|
gFilterBits = 0;
|
|
nFilterBits = 0;
|
|
wdogBits = 0;
|
|
|
|
pthAlertRunning = PI_THREAD_NONE;
|
|
pthFifoRunning = PI_THREAD_NONE;
|
|
pthSocketRunning = PI_THREAD_NONE;
|
|
|
|
wfc[0] = 0;
|
|
wfc[1] = 0;
|
|
wfc[2] = 0;
|
|
|
|
wfcur=0;
|
|
|
|
wfStats.micros = 0;
|
|
wfStats.highMicros = 0;
|
|
wfStats.maxMicros = PI_WAVE_MAX_MICROS;
|
|
|
|
wfStats.pulses = 0;
|
|
wfStats.highPulses = 0;
|
|
wfStats.maxPulses = PI_WAVE_MAX_PULSES;
|
|
|
|
wfStats.cbs = 0;
|
|
wfStats.highCbs = 0;
|
|
wfStats.maxCbs = (PI_WAVE_BLOCKS * PAGES_PER_BLOCK * CBS_PER_OPAGE);
|
|
|
|
gpioGetSamples.func = NULL;
|
|
gpioGetSamples.ex = 0;
|
|
gpioGetSamples.userdata = NULL;
|
|
gpioGetSamples.bits = 0;
|
|
|
|
for (i=0; i<=PI_MAX_USER_GPIO; i++)
|
|
{
|
|
wfRx[i].mode = PI_WFRX_NONE;
|
|
pthread_mutex_init(&wfRx[i].mutex, NULL);
|
|
gpioAlert[i].func = NULL;
|
|
}
|
|
|
|
for (i=0; i<=PI_MAX_GPIO; i++)
|
|
{
|
|
gpioInfo [i].is = GPIO_UNDEFINED;
|
|
gpioInfo [i].width = 0;
|
|
gpioInfo [i].range = PI_DEFAULT_DUTYCYCLE_RANGE;
|
|
gpioInfo [i].freqIdx = DEFAULT_PWM_IDX;
|
|
}
|
|
|
|
for (i=0; i<PI_NOTIFY_SLOTS; i++)
|
|
{
|
|
gpioNotify[i].seqno = 0;
|
|
gpioNotify[i].state = PI_NOTIFY_CLOSED;
|
|
}
|
|
|
|
for (i=0; i<=PI_MAX_SIGNUM; i++)
|
|
{
|
|
gpioSignal[i].func = NULL;
|
|
gpioSignal[i].ex = 0;
|
|
gpioSignal[i].userdata = NULL;
|
|
}
|
|
|
|
for (i=0; i<=PI_MAX_TIMER; i++)
|
|
{
|
|
gpioTimer[i].running = 0;
|
|
gpioTimer[i].func = NULL;
|
|
}
|
|
|
|
for (i=0; i<=PI_MAX_EVENT; i++)
|
|
{
|
|
eventAlert[i].func = NULL;
|
|
eventAlert[i].ignore = 0;
|
|
eventAlert[i].fired = 0;
|
|
}
|
|
|
|
/* calculate the usable PWM frequencies */
|
|
|
|
for (i=0; i<PWM_FREQS; i++)
|
|
{
|
|
pwmFreq[i]=
|
|
(1000000.0/
|
|
((float)PULSE_PER_CYCLE*gpioCfg.clockMicros*pwmCycles[i]))+0.5;
|
|
|
|
DBG(DBG_STARTUP, "f%d is %d", i, pwmFreq[i]);
|
|
}
|
|
|
|
inpFifo = NULL;
|
|
outFifo = NULL;
|
|
|
|
fdLock = -1;
|
|
fdMem = -1;
|
|
fdSock = -1;
|
|
|
|
dmaMboxBlk = MAP_FAILED;
|
|
dmaPMapBlk = MAP_FAILED;
|
|
dmaVirt = MAP_FAILED;
|
|
dmaBus = MAP_FAILED;
|
|
|
|
auxReg = MAP_FAILED;
|
|
clkReg = MAP_FAILED;
|
|
dmaReg = MAP_FAILED;
|
|
gpioReg = MAP_FAILED;
|
|
pcmReg = MAP_FAILED;
|
|
pwmReg = MAP_FAILED;
|
|
systReg = MAP_FAILED;
|
|
spiReg = MAP_FAILED;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static void initReleaseResources(void)
|
|
{
|
|
int i;
|
|
|
|
DBG(DBG_STARTUP, "");
|
|
|
|
/* shut down running threads */
|
|
|
|
for (i=0; i<=PI_MAX_GPIO; i++)
|
|
{
|
|
if (gpioISR[i].pth)
|
|
{
|
|
/* destroy thread, unexport GPIO */
|
|
|
|
gpioSetISRFunc(i, 0, 0, NULL);
|
|
}
|
|
}
|
|
|
|
for (i=0; i<=PI_MAX_TIMER; i++)
|
|
{
|
|
if (gpioTimer[i].running)
|
|
{
|
|
/* destroy thread */
|
|
|
|
pthread_cancel(gpioTimer[i].pthId);
|
|
pthread_join(gpioTimer[i].pthId, NULL);
|
|
gpioTimer[i].running = 0;
|
|
}
|
|
}
|
|
|
|
if (pthAlertRunning != PI_THREAD_NONE)
|
|
{
|
|
pthread_cancel(pthAlert);
|
|
pthread_join(pthAlert, NULL);
|
|
pthAlertRunning = PI_THREAD_NONE;
|
|
}
|
|
|
|
if (pthFifoRunning != PI_THREAD_NONE)
|
|
{
|
|
pthread_cancel(pthFifo);
|
|
pthread_join(pthFifo, NULL);
|
|
pthFifoRunning = PI_THREAD_NONE;
|
|
}
|
|
|
|
if (pthSocketRunning != PI_THREAD_NONE)
|
|
{
|
|
pthread_cancel(pthSocket);
|
|
pthread_join(pthSocket, NULL);
|
|
pthSocketRunning = PI_THREAD_NONE;
|
|
}
|
|
|
|
/* release mmap'd memory */
|
|
|
|
if (auxReg != MAP_FAILED) munmap((void *)auxReg, AUX_LEN);
|
|
if (bscsReg != MAP_FAILED) munmap((void *)bscsReg, BSCS_LEN);
|
|
if (clkReg != MAP_FAILED) munmap((void *)clkReg, CLK_LEN);
|
|
if (dmaReg != MAP_FAILED) munmap((void *)dmaReg, DMA_LEN);
|
|
if (gpioReg != MAP_FAILED) munmap((void *)gpioReg, GPIO_LEN);
|
|
if (pcmReg != MAP_FAILED) munmap((void *)pcmReg, PCM_LEN);
|
|
if (pwmReg != MAP_FAILED) munmap((void *)pwmReg, PWM_LEN);
|
|
if (systReg != MAP_FAILED) munmap((void *)systReg, SYST_LEN);
|
|
if (spiReg != MAP_FAILED) munmap((void *)spiReg, SPI_LEN);
|
|
|
|
auxReg = MAP_FAILED;
|
|
bscsReg = MAP_FAILED;
|
|
clkReg = MAP_FAILED;
|
|
dmaReg = MAP_FAILED;
|
|
gpioReg = MAP_FAILED;
|
|
pcmReg = MAP_FAILED;
|
|
pwmReg = MAP_FAILED;
|
|
systReg = MAP_FAILED;
|
|
spiReg = MAP_FAILED;
|
|
|
|
if (dmaBus != MAP_FAILED)
|
|
{
|
|
munmap(dmaBus,
|
|
PAGES_PER_BLOCK*(bufferBlocks+PI_WAVE_BLOCKS)*sizeof(dmaPage_t *));
|
|
}
|
|
|
|
dmaBus = MAP_FAILED;
|
|
|
|
if (dmaVirt != MAP_FAILED)
|
|
{
|
|
for (i=0; i<PAGES_PER_BLOCK*(bufferBlocks+PI_WAVE_BLOCKS); i++)
|
|
{
|
|
munmap(dmaVirt[i], PAGE_SIZE);
|
|
}
|
|
|
|
munmap(dmaVirt,
|
|
PAGES_PER_BLOCK*(bufferBlocks+PI_WAVE_BLOCKS)*sizeof(dmaPage_t *));
|
|
}
|
|
|
|
dmaVirt = MAP_FAILED;
|
|
|
|
if (dmaPMapBlk != MAP_FAILED)
|
|
{
|
|
for (i=0; i<(bufferBlocks+PI_WAVE_BLOCKS); i++)
|
|
{
|
|
munmap(dmaPMapBlk[i], PAGES_PER_BLOCK*PAGE_SIZE);
|
|
}
|
|
|
|
munmap(dmaPMapBlk, (bufferBlocks+PI_WAVE_BLOCKS)*sizeof(dmaPage_t *));
|
|
}
|
|
|
|
dmaPMapBlk = MAP_FAILED;
|
|
|
|
if (dmaMboxBlk != MAP_FAILED)
|
|
{
|
|
fdMbox = mbOpen();
|
|
|
|
for (i=0; i<(bufferBlocks+PI_WAVE_BLOCKS); i++)
|
|
{
|
|
mbDMAFree(&dmaMboxBlk[bufferBlocks+PI_WAVE_BLOCKS-i-1]);
|
|
}
|
|
|
|
mbClose(fdMbox);
|
|
|
|
munmap(dmaMboxBlk, (bufferBlocks+PI_WAVE_BLOCKS)*sizeof(DMAMem_t));
|
|
}
|
|
|
|
dmaMboxBlk = MAP_FAILED;
|
|
|
|
if (inpFifo != NULL)
|
|
{
|
|
fclose(inpFifo);
|
|
unlink(PI_INPFIFO);
|
|
inpFifo = NULL;
|
|
}
|
|
|
|
if (outFifo != NULL)
|
|
{
|
|
fclose(outFifo);
|
|
unlink(PI_OUTFIFO);
|
|
outFifo = NULL;
|
|
}
|
|
|
|
if (fdMem != -1)
|
|
{
|
|
close(fdMem);
|
|
fdMem = -1;
|
|
}
|
|
|
|
if (fdLock != -1)
|
|
{
|
|
close(fdLock);
|
|
unlink(PI_LOCKFILE);
|
|
fdLock = -1;
|
|
}
|
|
|
|
if (fdSock != -1)
|
|
{
|
|
close(fdSock);
|
|
fdSock = -1;
|
|
}
|
|
|
|
if (fdPmap != -1)
|
|
{
|
|
close(fdPmap);
|
|
fdPmap = -1;
|
|
}
|
|
|
|
if (fdMbox != -1)
|
|
{
|
|
close(fdMbox);
|
|
fdMbox = -1;
|
|
}
|
|
|
|
gpioStats.DMARestarts = 0;
|
|
gpioStats.dmaInitCbsCount = 0;
|
|
|
|
numSockNetAddr = 0;
|
|
}
|
|
|
|
int initInitialise(void)
|
|
{
|
|
int i;
|
|
unsigned rev, model;
|
|
struct sockaddr_in server;
|
|
struct sockaddr_in6 server6;
|
|
char * portStr;
|
|
unsigned port;
|
|
struct sched_param param;
|
|
pthread_attr_t pthAttr;
|
|
|
|
DBG(DBG_STARTUP, "");
|
|
|
|
waveClockInited = 0;
|
|
PWMClockInited = 0;
|
|
|
|
clock_gettime(CLOCK_REALTIME, &libStarted);
|
|
|
|
rev = gpioHardwareRevision();
|
|
|
|
initClearGlobals();
|
|
|
|
if (initCheckPermitted() < 0) return PI_INIT_FAILED;
|
|
|
|
fdLock = initGrabLockFile();
|
|
|
|
if (fdLock < 0)
|
|
SOFT_ERROR(PI_INIT_FAILED, "Can't lock %s", PI_LOCKFILE);
|
|
|
|
if (!gpioMaskSet)
|
|
{
|
|
if (rev == 0) gpioMask = PI_DEFAULT_UPDATE_MASK_UNKNOWN;
|
|
else if (rev < 4) gpioMask = PI_DEFAULT_UPDATE_MASK_B1;
|
|
else if (rev < 16) gpioMask = PI_DEFAULT_UPDATE_MASK_A_B2;
|
|
else if (rev == 17) gpioMask = PI_DEFAULT_UPDATE_MASK_COMPUTE;
|
|
else if (rev < 20) gpioMask = PI_DEFAULT_UPDATE_MASK_APLUS_BPLUS;
|
|
else if (rev == 20) gpioMask = PI_DEFAULT_UPDATE_MASK_COMPUTE;
|
|
else if (rev == 21) gpioMask = PI_DEFAULT_UPDATE_MASK_APLUS_BPLUS;
|
|
else
|
|
{
|
|
model = (rev >> 4) & 0xFF;
|
|
|
|
/* model
|
|
0=A 1=B
|
|
2=A+ 3=B+
|
|
4=Pi2B
|
|
5=Alpha
|
|
6=Compute Module
|
|
7=Unknown
|
|
8=Pi3B
|
|
9=Zero
|
|
12=Zero W
|
|
13=Pi3B+
|
|
14=Pi3A+
|
|
17=Pi4B
|
|
*/
|
|
if (model < 2) gpioMask = PI_DEFAULT_UPDATE_MASK_A_B2;
|
|
else if (model < 4) gpioMask = PI_DEFAULT_UPDATE_MASK_APLUS_BPLUS;
|
|
else if (model == 4) gpioMask = PI_DEFAULT_UPDATE_MASK_PI2B;
|
|
|
|
else if (model == 6
|
|
|| model ==10
|
|
|| model ==16) gpioMask = PI_DEFAULT_UPDATE_MASK_COMPUTE;
|
|
|
|
else if (model == 8
|
|
|| model ==13
|
|
|| model ==14) gpioMask = PI_DEFAULT_UPDATE_MASK_PI3B;
|
|
|
|
else if (model == 9
|
|
|| model ==12) gpioMask = PI_DEFAULT_UPDATE_MASK_ZERO;
|
|
|
|
else if (model ==17) gpioMask = PI_DEFAULT_UPDATE_MASK_PI4B;
|
|
else gpioMask = PI_DEFAULT_UPDATE_MASK_UNKNOWN;
|
|
}
|
|
|
|
gpioMaskSet = 1;
|
|
}
|
|
|
|
#ifndef EMBEDDED_IN_VM
|
|
if (!(gpioCfg.internals & PI_CFG_NOSIGHANDLER))
|
|
sigSetHandler();
|
|
#endif
|
|
|
|
if (initPeripherals() < 0) return PI_INIT_FAILED;
|
|
|
|
if (initAllocDMAMem() < 0) return PI_INIT_FAILED;
|
|
|
|
/* done with /dev/mem */
|
|
|
|
if (fdMem != -1)
|
|
{
|
|
close(fdMem);
|
|
fdMem = -1;
|
|
}
|
|
|
|
param.sched_priority = sched_get_priority_max(SCHED_FIFO);
|
|
|
|
if (gpioCfg.internals & PI_CFG_RT_PRIORITY)
|
|
sched_setscheduler(0, SCHED_FIFO, ¶m);
|
|
|
|
initClock(1); /* initialise main clock */
|
|
|
|
atexit(gpioTerminate);
|
|
|
|
if (pthread_attr_init(&pthAttr))
|
|
SOFT_ERROR(PI_INIT_FAILED, "pthread_attr_init failed (%m)");
|
|
|
|
if (pthread_attr_setstacksize(&pthAttr, STACK_SIZE))
|
|
SOFT_ERROR(PI_INIT_FAILED, "pthread_attr_setstacksize failed (%m)");
|
|
|
|
if (!(gpioCfg.ifFlags & PI_DISABLE_ALERT))
|
|
{
|
|
if (pthread_create(&pthAlert, &pthAttr, pthAlertThread, &i))
|
|
SOFT_ERROR(PI_INIT_FAILED, "pthread_create alert failed (%m)");
|
|
|
|
pthAlertRunning = PI_THREAD_STARTED;
|
|
}
|
|
|
|
if (!(gpioCfg.ifFlags & PI_DISABLE_FIFO_IF))
|
|
{
|
|
if (pthread_create(&pthFifo, &pthAttr, pthFifoThread, &i))
|
|
SOFT_ERROR(PI_INIT_FAILED, "pthread_create fifo failed (%m)");
|
|
|
|
pthFifoRunning = PI_THREAD_STARTED;
|
|
}
|
|
|
|
if (!(gpioCfg.ifFlags & PI_DISABLE_SOCK_IF))
|
|
{
|
|
portStr = getenv(PI_ENVPORT);
|
|
if (portStr) port = atoi(portStr); else port = gpioCfg.socketPort;
|
|
|
|
// Accept connections on IPv6, unless we have an IPv4-only whitelist
|
|
if (!numSockNetAddr)
|
|
{
|
|
fdSock = socket(AF_INET6, SOCK_STREAM , 0);
|
|
|
|
if (fdSock != -1)
|
|
{
|
|
bzero((char *)&server6, sizeof(server6));
|
|
server6.sin6_family = AF_INET6;
|
|
if (gpioCfg.ifFlags & PI_LOCALHOST_SOCK_IF)
|
|
{
|
|
server6.sin6_addr = in6addr_loopback;
|
|
}
|
|
else
|
|
{
|
|
server6.sin6_addr = in6addr_any;
|
|
}
|
|
server6.sin6_port = htons(port);
|
|
|
|
int opt = 1;
|
|
setsockopt(fdSock, SOL_SOCKET, SO_REUSEADDR, &opt, sizeof(opt));
|
|
if (bind(fdSock,(struct sockaddr *)&server6, sizeof(server6)) < 0)
|
|
SOFT_ERROR(PI_INIT_FAILED, "bind to port %d failed (%m)", port);
|
|
}
|
|
}
|
|
|
|
if (numSockNetAddr || fdSock == -1)
|
|
{
|
|
fdSock = socket(AF_INET , SOCK_STREAM , 0);
|
|
|
|
if (fdSock == -1)
|
|
SOFT_ERROR(PI_INIT_FAILED, "socket failed (%m)");
|
|
else
|
|
{
|
|
int opt = 1;
|
|
setsockopt(fdSock, SOL_SOCKET, SO_REUSEADDR, &opt, sizeof(opt));
|
|
}
|
|
server.sin_family = AF_INET;
|
|
if (gpioCfg.ifFlags & PI_LOCALHOST_SOCK_IF)
|
|
{
|
|
server.sin_addr.s_addr = htonl(INADDR_LOOPBACK);
|
|
}
|
|
else
|
|
{
|
|
server.sin_addr.s_addr = htonl(INADDR_ANY);
|
|
}
|
|
server.sin_port = htons(port);
|
|
|
|
if (bind(fdSock,(struct sockaddr *)&server , sizeof(server)) < 0)
|
|
SOFT_ERROR(PI_INIT_FAILED, "bind to port %d failed (%m)", port);
|
|
}
|
|
|
|
if (pthread_create(&pthSocket, &pthAttr, pthSocketThread, &i))
|
|
SOFT_ERROR(PI_INIT_FAILED, "pthread_create socket failed (%m)");
|
|
|
|
pthSocketRunning = PI_THREAD_STARTED;
|
|
}
|
|
|
|
myGpioDelay(1000);
|
|
|
|
dmaInitCbs();
|
|
|
|
flushMemory();
|
|
|
|
//cast twice to suppress compiler warning, I belive this cast
|
|
//is ok because dmaIBus contains bus addresses, not virtual
|
|
//addresses.
|
|
initDMAgo((uint32_t *)dmaIn, (uint32_t)(uintptr_t)dmaIBus[0]);
|
|
|
|
return PIGPIO_VERSION;
|
|
}
|
|
|
|
|
|
/* ======================================================================= */
|
|
|
|
int getBitInBytes(int bitPos, char *buf, int numBits)
|
|
{
|
|
int bitp, bufp;
|
|
|
|
if (bitPos < numBits)
|
|
{
|
|
bufp = bitPos / 8;
|
|
bitp = 7 - (bitPos % 8);
|
|
if (buf[bufp] & (1<<bitp)) return 1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
void putBitInBytes(int bitPos, char *buf, int bit)
|
|
{
|
|
int bitp, bufp;
|
|
|
|
bufp = bitPos / 8;
|
|
bitp = 7 - (bitPos % 8);
|
|
|
|
if (bit) buf[bufp] |= (1<<bitp);
|
|
else buf[bufp] &= (~(1<<bitp));
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
uint32_t rawWaveGetOOL(int pos)
|
|
{
|
|
int page, slot;
|
|
|
|
if ((pos >= 0) && (pos < NUM_WAVE_OOL))
|
|
{
|
|
waveOOLPageSlot(pos, &page, &slot);
|
|
return (dmaOVirt[page]->OOL[slot]);
|
|
}
|
|
|
|
return -1;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
void rawWaveSetOOL(int pos, uint32_t value)
|
|
{
|
|
int page, slot;
|
|
|
|
if ((pos >= 0) && (pos < NUM_WAVE_OOL))
|
|
{
|
|
waveOOLPageSlot(pos, &page, &slot);
|
|
dmaOVirt[page]->OOL[slot] = value;
|
|
}
|
|
}
|
|
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
uint32_t rawWaveGetOut(int pos)
|
|
{
|
|
int page, slot;
|
|
|
|
if ((pos >= 0) && (pos < NUM_WAVE_OOL))
|
|
{
|
|
waveOOLPageSlot(pos, &page, &slot);
|
|
return (dmaOVirt[page]->OOL[slot]);
|
|
}
|
|
|
|
return -1;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
void rawWaveSetOut(int pos, uint32_t value)
|
|
{
|
|
int page, slot;
|
|
|
|
if ((pos >= 0) && (pos < NUM_WAVE_OOL))
|
|
{
|
|
waveOOLPageSlot(pos, &page, &slot);
|
|
dmaOVirt[page]->OOL[slot] = value;
|
|
}
|
|
}
|
|
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
uint32_t rawWaveGetIn(int pos)
|
|
{
|
|
int page, slot;
|
|
|
|
if ((pos >= 0) && (pos < NUM_WAVE_OOL))
|
|
{
|
|
waveOOLPageSlot((NUM_WAVE_OOL-1)-pos, &page, &slot);
|
|
return (dmaOVirt[page]->OOL[slot]);
|
|
}
|
|
|
|
return -1;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
void rawWaveSetIn(int pos, uint32_t value)
|
|
{
|
|
int page, slot;
|
|
|
|
if ((pos >= 0) && (pos < NUM_WAVE_OOL))
|
|
{
|
|
waveOOLPageSlot((NUM_WAVE_OOL-1)-pos, &page, &slot);
|
|
dmaOVirt[page]->OOL[slot] = value;
|
|
}
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
rawWaveInfo_t rawWaveInfo(int wave_id)
|
|
{
|
|
rawWaveInfo_t dummy = {0, 0, 0, 0, 0, 0, 0, 0};
|
|
|
|
if ((wave_id >=0) && (wave_id < PI_MAX_WAVES)) return waveInfo[wave_id];
|
|
else return dummy;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
double time_time(void)
|
|
{
|
|
struct timeval tv;
|
|
double t;
|
|
|
|
gettimeofday(&tv, 0);
|
|
|
|
t = (double)tv.tv_sec + ((double)tv.tv_usec / 1E6);
|
|
|
|
return t;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
void time_sleep(double seconds)
|
|
{
|
|
struct timespec ts, rem;
|
|
|
|
if (seconds > 0.0)
|
|
{
|
|
ts.tv_sec = seconds;
|
|
ts.tv_nsec = (seconds-(double)ts.tv_sec) * 1E9;
|
|
|
|
while (clock_nanosleep(CLOCK_REALTIME, 0, &ts, &rem))
|
|
{
|
|
/* copy remaining time to ts */
|
|
ts.tv_sec = rem.tv_sec;
|
|
ts.tv_nsec = rem.tv_nsec;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
void rawDumpWave(void)
|
|
{
|
|
int i;
|
|
|
|
unsigned numWaves, t;
|
|
|
|
rawWave_t *waves;
|
|
|
|
numWaves = wfc[wfcur];
|
|
waves = wf [wfcur];
|
|
|
|
t = 0;
|
|
|
|
for (i=0; i<numWaves; i++)
|
|
{
|
|
fprintf(stderr, "%10u %08X %08X %08X %10u\n",
|
|
t, waves[i].gpioOn, waves[i].gpioOff,
|
|
waves[i].flags, waves[i].usDelay);
|
|
t += waves[i].usDelay;
|
|
}
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
void rawDumpScript(unsigned script_id)
|
|
{
|
|
int i;
|
|
|
|
if (script_id >= PI_MAX_SCRIPTS) return;
|
|
|
|
if (gpioScript[script_id].state == PI_SCRIPT_IN_USE)
|
|
{
|
|
|
|
for (i=0; i<PI_MAX_SCRIPT_PARAMS; i++)
|
|
{
|
|
fprintf(stderr, "p%d=%d ", i, gpioScript[script_id].script.par[i]);
|
|
}
|
|
|
|
fprintf(stderr, "\n");
|
|
|
|
for (i=0; i<PI_MAX_SCRIPT_VARS; i++)
|
|
{
|
|
fprintf(stderr, "v%d=%d ", i, gpioScript[script_id].script.var[i]);
|
|
}
|
|
|
|
fprintf(stderr, "\n");
|
|
|
|
for (i=0; i<gpioScript[script_id].script.instrs; i++)
|
|
{
|
|
fprintf(stderr,
|
|
"c%d=[%"PRIdPTR", %"PRIdPTR"(%d), %"PRIdPTR"(%d), %"PRIdPTR", %"PRIdPTR"]\n",
|
|
i,
|
|
gpioScript[script_id].script.instr[i].p[0],
|
|
gpioScript[script_id].script.instr[i].p[1],
|
|
gpioScript[script_id].script.instr[i].opt[1],
|
|
gpioScript[script_id].script.instr[i].p[2],
|
|
gpioScript[script_id].script.instr[i].opt[2],
|
|
gpioScript[script_id].script.instr[i].p[3],
|
|
gpioScript[script_id].script.instr[i].p[4]);
|
|
}
|
|
}
|
|
}
|
|
|
|
/* ======================================================================= */
|
|
|
|
int gpioInitialise(void)
|
|
{
|
|
int status;
|
|
|
|
if (libInitialised) return PIGPIO_VERSION;
|
|
|
|
DBG(DBG_STARTUP, "not initialised, initialising");
|
|
|
|
runState = PI_STARTING;
|
|
|
|
status = initInitialise();
|
|
|
|
if (status < 0)
|
|
{
|
|
runState = PI_ENDING;
|
|
initReleaseResources();
|
|
}
|
|
else
|
|
{
|
|
libInitialised = 1;
|
|
|
|
runState = PI_RUNNING;
|
|
|
|
if (!(gpioCfg.ifFlags & PI_DISABLE_ALERT))
|
|
{
|
|
while (pthAlertRunning != PI_THREAD_RUNNING) myGpioDelay(1000);
|
|
}
|
|
|
|
}
|
|
|
|
return status;
|
|
}
|
|
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
void gpioTerminate(void)
|
|
{
|
|
int i;
|
|
|
|
DBG(DBG_USER, "");
|
|
|
|
if (!libInitialised) return;
|
|
|
|
DBG(DBG_STARTUP, "initialised, terminating");
|
|
|
|
runState = PI_ENDING;
|
|
|
|
gpioMaskSet = 0;
|
|
|
|
/* reset DMA */
|
|
|
|
if (dmaReg != MAP_FAILED)
|
|
{
|
|
initKillDMA(dmaIn);
|
|
initKillDMA(dmaOut);
|
|
}
|
|
|
|
#ifndef EMBEDDED_IN_VM
|
|
if ((gpioCfg.internals & PI_CFG_STATS) &&
|
|
(!(gpioCfg.internals & PI_CFG_NOSIGHANDLER)))
|
|
{
|
|
fprintf(stderr,
|
|
"\n#####################################################\n");
|
|
fprintf(stderr, "pigpio version=%d internals=%X\n",
|
|
PIGPIO_VERSION, gpioCfg.internals);
|
|
|
|
fprintf(stderr,
|
|
"micros=%d allocMode=%d dmaInitCbs=%d DMARestarts=%d\n",
|
|
gpioCfg.clockMicros, gpioCfg.memAllocMode,
|
|
gpioStats.dmaInitCbsCount, gpioStats.DMARestarts);
|
|
|
|
fprintf(stderr,
|
|
"samples %u maxSamples %u maxEmit %u emitFrags %u\n",
|
|
gpioStats.numSamples, gpioStats.maxSamples,
|
|
gpioStats.maxEmit, gpioStats.emitFrags);
|
|
|
|
fprintf(stderr, "cbTicks %d, cbCalls %u\n",
|
|
gpioStats.cbTicks, gpioStats.cbCalls);
|
|
|
|
fprintf(stderr, "pipe: good %u, short %u, would block %u\n",
|
|
gpioStats.goodPipeWrite, gpioStats.shortPipeWrite,
|
|
gpioStats.wouldBlockPipeWrite);
|
|
|
|
fprintf(stderr, "alertTicks %u, lateTicks %u, moreToDo %u\n",
|
|
gpioStats.alertTicks, gpioStats.lateTicks, gpioStats.moreToDo);
|
|
|
|
for (i=0; i< TICKSLOTS; i++)
|
|
fprintf(stderr, "%9u ", gpioStats.diffTick[i]);
|
|
|
|
fprintf(stderr,
|
|
"\n#####################################################\n\n\n");
|
|
}
|
|
|
|
#endif
|
|
initReleaseResources();
|
|
|
|
fflush(NULL);
|
|
|
|
libInitialised = 0;
|
|
}
|
|
|
|
static void switchFunctionOff(unsigned gpio)
|
|
{
|
|
switch (gpioInfo[gpio].is)
|
|
{
|
|
case GPIO_SERVO:
|
|
/* switch servo off */
|
|
myGpioSetServo(gpio, gpioInfo[gpio].width, 0);
|
|
gpioInfo[gpio].width = 0;
|
|
break;
|
|
|
|
case GPIO_PWM:
|
|
/* switch pwm off */
|
|
myGpioSetPwm(gpio, gpioInfo[gpio].width, 0);
|
|
gpioInfo[gpio].width = 0;
|
|
break;
|
|
|
|
case GPIO_HW_CLK:
|
|
/* No longer disable clock hardware, doing that was a bug. */
|
|
gpioInfo[gpio].width = 0;
|
|
break;
|
|
|
|
case GPIO_HW_PWM:
|
|
/* No longer disable PWM hardware, doing that was a bug. */
|
|
gpioInfo[gpio].width = 0;
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void stopHardwarePWM(void)
|
|
{
|
|
unsigned i, pwm;
|
|
|
|
for (i=0; i<= PI_MAX_GPIO; i++)
|
|
{
|
|
if (gpioInfo[i].is == GPIO_HW_PWM)
|
|
{
|
|
pwm = (PWMDef[i] >> 4) & 3;
|
|
|
|
if (pwm == 0) pwmReg[PWM_CTL] &= (~PWM_CTL_PWEN1);
|
|
else pwmReg[PWM_CTL] &= (~PWM_CTL_PWEN2);
|
|
|
|
gpioInfo[i].width = 0;
|
|
gpioInfo[i].is = GPIO_UNDEFINED;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioSetMode(unsigned gpio, unsigned mode)
|
|
{
|
|
int reg, shift, old_mode;
|
|
static pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER;
|
|
|
|
DBG(DBG_USER, "gpio=%d mode=%d", gpio, mode);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (gpio > PI_MAX_GPIO)
|
|
SOFT_ERROR(PI_BAD_GPIO, "bad gpio (%d)", gpio);
|
|
|
|
if (mode > PI_ALT3)
|
|
SOFT_ERROR(PI_BAD_MODE, "gpio %d, bad mode (%d)", gpio, mode);
|
|
|
|
pthread_mutex_lock(&mutex);
|
|
|
|
reg = gpio/10;
|
|
shift = (gpio%10) * 3;
|
|
|
|
old_mode = (gpioReg[reg] >> shift) & 7;
|
|
|
|
if (mode != old_mode)
|
|
{
|
|
switchFunctionOff(gpio);
|
|
|
|
gpioInfo[gpio].is = GPIO_UNDEFINED;
|
|
}
|
|
|
|
gpioReg[reg] = (gpioReg[reg] & ~(7<<shift)) | (mode<<shift);
|
|
|
|
pthread_mutex_unlock(&mutex);
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioGetMode(unsigned gpio)
|
|
{
|
|
int reg, shift;
|
|
|
|
DBG(DBG_USER, "gpio=%d", gpio);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (gpio > PI_MAX_GPIO)
|
|
SOFT_ERROR(PI_BAD_GPIO, "bad gpio (%d)", gpio);
|
|
|
|
reg = gpio/10;
|
|
shift = (gpio%10) * 3;
|
|
|
|
return (gpioReg[reg] >> shift) & 7;
|
|
}
|
|
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioSetPullUpDown(unsigned gpio, unsigned pud)
|
|
{
|
|
int shift = (gpio & 0xf) << 1;
|
|
uint32_t bits;
|
|
uint32_t pull;
|
|
|
|
DBG(DBG_USER, "gpio=%d pud=%d", gpio, pud);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (gpio > PI_MAX_GPIO)
|
|
SOFT_ERROR(PI_BAD_GPIO, "bad gpio (%d)", gpio);
|
|
|
|
if (pud > PI_PUD_UP)
|
|
SOFT_ERROR(PI_BAD_PUD, "gpio %d, bad pud (%d)", gpio, pud);
|
|
|
|
if (pi_is_2711)
|
|
{
|
|
switch (pud)
|
|
{
|
|
case PI_PUD_OFF: pull = 0; break;
|
|
case PI_PUD_UP: pull = 1; break;
|
|
case PI_PUD_DOWN: pull = 2; break;
|
|
}
|
|
|
|
bits = *(gpioReg + GPPUPPDN0 + (gpio>>4));
|
|
bits &= ~(3 << shift);
|
|
bits |= (pull << shift);
|
|
*(gpioReg + GPPUPPDN0 + (gpio>>4)) = bits;
|
|
}
|
|
else
|
|
{
|
|
*(gpioReg + GPPUD) = pud;
|
|
|
|
myGpioDelay(1);
|
|
|
|
*(gpioReg + GPPUDCLK0 + BANK) = BIT;
|
|
|
|
myGpioDelay(1);
|
|
|
|
*(gpioReg + GPPUD) = 0;
|
|
|
|
*(gpioReg + GPPUDCLK0 + BANK) = 0;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioRead(unsigned gpio)
|
|
{
|
|
DBG(DBG_USER, "gpio=%d", gpio);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (gpio > PI_MAX_GPIO)
|
|
SOFT_ERROR(PI_BAD_GPIO, "bad gpio (%d)", gpio);
|
|
|
|
if ((*(gpioReg + GPLEV0 + BANK) & BIT) != 0) return PI_ON;
|
|
else return PI_OFF;
|
|
}
|
|
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioWrite(unsigned gpio, unsigned level)
|
|
{
|
|
DBG(DBG_USER, "gpio=%d level=%d", gpio, level);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (gpio > PI_MAX_GPIO)
|
|
SOFT_ERROR(PI_BAD_GPIO, "bad gpio (%d)", gpio);
|
|
|
|
if (level > PI_ON)
|
|
SOFT_ERROR(PI_BAD_LEVEL, "gpio %d, bad level (%d)", gpio, level);
|
|
|
|
if (gpio <= PI_MAX_GPIO)
|
|
{
|
|
if (gpioInfo[gpio].is != GPIO_WRITE)
|
|
{
|
|
/* stop a glitch between setting mode then level */
|
|
if (level == PI_OFF) *(gpioReg + GPCLR0 + BANK) = BIT;
|
|
else *(gpioReg + GPSET0 + BANK) = BIT;
|
|
|
|
switchFunctionOff(gpio);
|
|
|
|
gpioInfo[gpio].is = GPIO_WRITE;
|
|
}
|
|
}
|
|
|
|
myGpioSetMode(gpio, PI_OUTPUT);
|
|
|
|
if (level == PI_OFF) *(gpioReg + GPCLR0 + BANK) = BIT;
|
|
else *(gpioReg + GPSET0 + BANK) = BIT;
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioPWM(unsigned gpio, unsigned val)
|
|
{
|
|
DBG(DBG_USER, "gpio=%d dutycycle=%d", gpio, val);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (gpio > PI_MAX_USER_GPIO)
|
|
SOFT_ERROR(PI_BAD_USER_GPIO, "bad gpio (%d)", gpio);
|
|
|
|
if (val > gpioInfo[gpio].range)
|
|
SOFT_ERROR(PI_BAD_DUTYCYCLE, "gpio %d, bad dutycycle (%d)", gpio, val);
|
|
|
|
if (gpioInfo[gpio].is != GPIO_PWM)
|
|
{
|
|
switchFunctionOff(gpio);
|
|
|
|
gpioInfo[gpio].is = GPIO_PWM;
|
|
|
|
if (!val) myGpioWrite(gpio, 0);
|
|
}
|
|
|
|
myGpioSetMode(gpio, PI_OUTPUT);
|
|
|
|
myGpioSetPwm(gpio, gpioInfo[gpio].width, val);
|
|
|
|
gpioInfo[gpio].width=val;
|
|
|
|
return 0;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioGetPWMdutycycle(unsigned gpio)
|
|
{
|
|
unsigned pwm;
|
|
|
|
DBG(DBG_USER, "gpio=%d", gpio);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (gpio > PI_MAX_USER_GPIO)
|
|
SOFT_ERROR(PI_BAD_USER_GPIO, "bad gpio (%d)", gpio);
|
|
|
|
switch (gpioInfo[gpio].is)
|
|
{
|
|
case GPIO_PWM:
|
|
return gpioInfo[gpio].width;
|
|
|
|
case GPIO_HW_PWM:
|
|
pwm = (PWMDef[gpio] >> 4) & 3;
|
|
return hw_pwm_duty[pwm];
|
|
|
|
case GPIO_HW_CLK:
|
|
return PI_HW_PWM_RANGE/2;
|
|
|
|
default:
|
|
SOFT_ERROR(PI_NOT_PWM_GPIO, "not a PWM gpio (%d)", gpio);
|
|
}
|
|
}
|
|
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioSetPWMrange(unsigned gpio, unsigned range)
|
|
{
|
|
int oldWidth, newWidth;
|
|
|
|
DBG(DBG_USER, "gpio=%d range=%d", gpio, range);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (gpio > PI_MAX_USER_GPIO)
|
|
SOFT_ERROR(PI_BAD_USER_GPIO, "bad gpio (%d)", gpio);
|
|
|
|
if ((range < PI_MIN_DUTYCYCLE_RANGE) || (range > PI_MAX_DUTYCYCLE_RANGE))
|
|
SOFT_ERROR(PI_BAD_DUTYRANGE, "gpio %d, bad range (%d)", gpio, range);
|
|
|
|
oldWidth = gpioInfo[gpio].width;
|
|
|
|
if (oldWidth)
|
|
{
|
|
if (gpioInfo[gpio].is == GPIO_PWM)
|
|
{
|
|
newWidth = (range * oldWidth) / gpioInfo[gpio].range;
|
|
|
|
myGpioSetPwm(gpio, oldWidth, 0);
|
|
gpioInfo[gpio].range = range;
|
|
gpioInfo[gpio].width = newWidth;
|
|
myGpioSetPwm(gpio, 0, newWidth);
|
|
}
|
|
}
|
|
|
|
gpioInfo[gpio].range = range;
|
|
|
|
/* return the actual range for the current gpio frequency */
|
|
|
|
return pwmRealRange[gpioInfo[gpio].freqIdx];
|
|
}
|
|
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioGetPWMrange(unsigned gpio)
|
|
{
|
|
DBG(DBG_USER, "gpio=%d", gpio);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (gpio > PI_MAX_USER_GPIO)
|
|
SOFT_ERROR(PI_BAD_USER_GPIO, "bad gpio (%d)", gpio);
|
|
|
|
switch (gpioInfo[gpio].is)
|
|
{
|
|
case GPIO_HW_PWM:
|
|
case GPIO_HW_CLK:
|
|
return PI_HW_PWM_RANGE;
|
|
|
|
default:
|
|
return gpioInfo[gpio].range;
|
|
}
|
|
}
|
|
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioGetPWMrealRange(unsigned gpio)
|
|
{
|
|
unsigned pwm;
|
|
|
|
DBG(DBG_USER, "gpio=%d", gpio);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (gpio > PI_MAX_USER_GPIO)
|
|
SOFT_ERROR(PI_BAD_USER_GPIO, "bad gpio (%d)", gpio);
|
|
|
|
switch (gpioInfo[gpio].is)
|
|
{
|
|
case GPIO_HW_PWM:
|
|
pwm = (PWMDef[gpio] >> 4) & 3;
|
|
return hw_pwm_real_range[pwm];
|
|
|
|
case GPIO_HW_CLK:
|
|
return PI_HW_PWM_RANGE;
|
|
|
|
default:
|
|
return pwmRealRange[gpioInfo[gpio].freqIdx];
|
|
}
|
|
}
|
|
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioSetPWMfrequency(unsigned gpio, unsigned frequency)
|
|
{
|
|
int i, width;
|
|
unsigned diff, best, idx;
|
|
|
|
DBG(DBG_USER, "gpio=%d frequency=%d", gpio, frequency);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (gpio > PI_MAX_USER_GPIO)
|
|
SOFT_ERROR(PI_BAD_USER_GPIO, "bad gpio (%d)", gpio);
|
|
|
|
if (frequency > pwmFreq[0]) idx = 0;
|
|
else if (frequency < pwmFreq[PWM_FREQS-1]) idx = PWM_FREQS-1;
|
|
else
|
|
{
|
|
best = 100000; /* impossibly high frequency difference */
|
|
idx = 0;
|
|
|
|
for (i=0; i<PWM_FREQS; i++)
|
|
{
|
|
if (frequency > pwmFreq[i]) diff = frequency - pwmFreq[i];
|
|
else diff = pwmFreq[i] - frequency;
|
|
|
|
if (diff < best)
|
|
{
|
|
best = diff;
|
|
idx = i;
|
|
}
|
|
}
|
|
}
|
|
|
|
width = gpioInfo[gpio].width;
|
|
|
|
if (width)
|
|
{
|
|
if (gpioInfo[gpio].is == GPIO_PWM)
|
|
{
|
|
myGpioSetPwm(gpio, width, 0);
|
|
gpioInfo[gpio].freqIdx = idx;
|
|
myGpioSetPwm(gpio, 0, width);
|
|
}
|
|
}
|
|
|
|
gpioInfo[gpio].freqIdx = idx;
|
|
|
|
return pwmFreq[idx];
|
|
}
|
|
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioGetPWMfrequency(unsigned gpio)
|
|
{
|
|
unsigned pwm, clock;
|
|
|
|
DBG(DBG_USER, "gpio=%d", gpio);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (gpio > PI_MAX_USER_GPIO)
|
|
SOFT_ERROR(PI_BAD_USER_GPIO, "bad gpio (%d)", gpio);
|
|
|
|
switch (gpioInfo[gpio].is)
|
|
{
|
|
case GPIO_HW_PWM:
|
|
pwm = (PWMDef[gpio] >> 4) & 3;
|
|
return hw_pwm_freq[pwm];
|
|
|
|
case GPIO_HW_CLK:
|
|
clock = (clkDef[gpio] >> 4) & 3;
|
|
return hw_clk_freq[clock];
|
|
|
|
default:
|
|
return pwmFreq[gpioInfo[gpio].freqIdx];
|
|
}
|
|
}
|
|
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioServo(unsigned gpio, unsigned val)
|
|
{
|
|
DBG(DBG_USER, "gpio=%d pulsewidth=%d", gpio, val);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (gpio > PI_MAX_USER_GPIO)
|
|
SOFT_ERROR(PI_BAD_USER_GPIO, "bad gpio (%d)", gpio);
|
|
|
|
if ((val!=PI_SERVO_OFF) && (val<PI_MIN_SERVO_PULSEWIDTH))
|
|
SOFT_ERROR(PI_BAD_PULSEWIDTH,
|
|
"gpio %d, bad pulsewidth (%d)", gpio, val);
|
|
|
|
if (val>PI_MAX_SERVO_PULSEWIDTH)
|
|
SOFT_ERROR(PI_BAD_PULSEWIDTH,
|
|
"gpio %d, bad pulsewidth (%d)", gpio, val);
|
|
|
|
if (gpioInfo[gpio].is != GPIO_SERVO)
|
|
{
|
|
switchFunctionOff(gpio);
|
|
|
|
gpioInfo[gpio].is = GPIO_SERVO;
|
|
|
|
if (!val) myGpioWrite(gpio, 0);
|
|
}
|
|
|
|
myGpioSetMode(gpio, PI_OUTPUT);
|
|
|
|
myGpioSetServo(gpio, gpioInfo[gpio].width, val);
|
|
|
|
gpioInfo[gpio].width=val;
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioGetServoPulsewidth(unsigned gpio)
|
|
{
|
|
DBG(DBG_USER, "gpio=%d", gpio);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (gpio > PI_MAX_USER_GPIO)
|
|
SOFT_ERROR(PI_BAD_USER_GPIO, "bad gpio (%d)", gpio);
|
|
|
|
if (gpioInfo[gpio].is != GPIO_SERVO)
|
|
SOFT_ERROR(PI_NOT_SERVO_GPIO, "not a servo gpio (%d)", gpio);
|
|
|
|
return gpioInfo[gpio].width;
|
|
}
|
|
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioWaveClear(void)
|
|
{
|
|
DBG(DBG_USER, "");
|
|
|
|
CHECK_INITED;
|
|
|
|
wfc[0] = 0;
|
|
wfc[1] = 0;
|
|
wfc[2] = 0;
|
|
|
|
wfcur = 0;
|
|
|
|
wfStats.micros = 0;
|
|
wfStats.pulses = 0;
|
|
wfStats.cbs = 0;
|
|
|
|
waveOutBotCB = PI_WAVE_COUNT_PAGES*CBS_PER_OPAGE;
|
|
waveOutBotOOL = PI_WAVE_COUNT_PAGES*OOL_PER_OPAGE;
|
|
waveOutTopOOL = NUM_WAVE_OOL;
|
|
|
|
waveOutCount = 0;
|
|
|
|
waveEndPtr = NULL;
|
|
|
|
return 0;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioWaveAddNew(void)
|
|
{
|
|
DBG(DBG_USER, "");
|
|
|
|
CHECK_INITED;
|
|
|
|
wfc[0] = 0;
|
|
wfc[1] = 0;
|
|
wfc[2] = 0;
|
|
|
|
wfcur = 0;
|
|
|
|
wfStats.micros = 0;
|
|
wfStats.pulses = 0;
|
|
wfStats.cbs = 0;
|
|
|
|
return 0;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioWaveAddGeneric(unsigned numPulses, gpioPulse_t *pulses)
|
|
{
|
|
int p;
|
|
|
|
DBG(DBG_USER, "numPulses=%u pulses=%08"PRIXPTR, numPulses, (uintptr_t)pulses);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (numPulses > PI_WAVE_MAX_PULSES)
|
|
SOFT_ERROR(PI_TOO_MANY_PULSES, "bad number of pulses (%d)", numPulses);
|
|
|
|
if (!pulses) SOFT_ERROR(PI_BAD_POINTER, "bad (NULL) pulses pointer");
|
|
|
|
for (p=0; p<numPulses; p++)
|
|
{
|
|
wf[2][p].gpioOff = pulses[p].gpioOff;
|
|
wf[2][p].gpioOn = pulses[p].gpioOn;
|
|
wf[2][p].usDelay = pulses[p].usDelay;
|
|
wf[2][p].flags = 0;
|
|
}
|
|
|
|
return rawWaveAddGeneric(numPulses, wf[2]);
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioWaveAddSerial
|
|
(unsigned gpio,
|
|
unsigned baud,
|
|
unsigned data_bits,
|
|
unsigned stop_bits,
|
|
unsigned offset,
|
|
unsigned numBytes,
|
|
char *bstr)
|
|
{
|
|
int i, b, p, lev, c, v;
|
|
|
|
uint16_t *wstr = (uint16_t *)bstr;
|
|
uint32_t *lstr = (uint32_t *)bstr;
|
|
|
|
unsigned bitDelay[32];
|
|
|
|
DBG(DBG_USER,
|
|
"gpio=%d baud=%d bits=%d stops=%d offset=%d numBytes=%d str=[%s]",
|
|
gpio, baud, data_bits, stop_bits, offset,
|
|
numBytes, myBuf2Str(numBytes, (char *)bstr));
|
|
|
|
CHECK_INITED;
|
|
|
|
if (gpio > PI_MAX_USER_GPIO)
|
|
SOFT_ERROR(PI_BAD_USER_GPIO, "bad gpio (%d)", gpio);
|
|
|
|
if ((baud < PI_WAVE_MIN_BAUD) || (baud > PI_WAVE_MAX_BAUD))
|
|
SOFT_ERROR(PI_BAD_WAVE_BAUD, "bad baud rate (%d)", baud);
|
|
|
|
if ((data_bits < PI_MIN_WAVE_DATABITS) ||
|
|
(data_bits > PI_MAX_WAVE_DATABITS))
|
|
SOFT_ERROR(PI_BAD_DATABITS, "bad number of databits (%d)", data_bits);
|
|
|
|
if ((stop_bits < PI_MIN_WAVE_HALFSTOPBITS) ||
|
|
(stop_bits > PI_MAX_WAVE_HALFSTOPBITS))
|
|
SOFT_ERROR(PI_BAD_STOPBITS,
|
|
"bad number of (half) stop bits (%d)", stop_bits);
|
|
|
|
if (gpio > PI_MAX_USER_GPIO)
|
|
SOFT_ERROR(PI_BAD_USER_GPIO, "bad gpio (%d)", gpio);
|
|
|
|
if (numBytes > PI_WAVE_MAX_CHARS)
|
|
SOFT_ERROR(PI_TOO_MANY_CHARS, "too many chars (%d)", numBytes);
|
|
|
|
if (offset > PI_WAVE_MAX_MICROS)
|
|
SOFT_ERROR(PI_BAD_SER_OFFSET, "offset too large (%d)", offset);
|
|
|
|
if (data_bits > 8) numBytes /= 2;
|
|
if (data_bits > 16) numBytes /= 2;
|
|
|
|
if (!numBytes) return 0;
|
|
|
|
waveBitDelay(baud, data_bits, stop_bits, bitDelay);
|
|
|
|
p = 0;
|
|
|
|
wf[2][p].gpioOn = (1<<gpio);
|
|
wf[2][p].gpioOff = 0;
|
|
wf[2][p].flags = 0;
|
|
|
|
if (offset > bitDelay[0]) wf[2][p].usDelay = offset;
|
|
else wf[2][p].usDelay = bitDelay[0];
|
|
|
|
for (i=0; i<numBytes; i++)
|
|
{
|
|
p++;
|
|
|
|
/* start bit */
|
|
|
|
wf[2][p].gpioOn = 0;
|
|
wf[2][p].gpioOff = (1<<gpio);
|
|
wf[2][p].usDelay = bitDelay[0];
|
|
wf[2][p].flags = 0;
|
|
|
|
lev = 0;
|
|
|
|
if (data_bits < 9) c = bstr[i];
|
|
else if (data_bits < 17) c = wstr[i];
|
|
else c = lstr[i];
|
|
|
|
for (b=0; b<data_bits; b++)
|
|
{
|
|
if (c & (1<<b)) v=1; else v=0;
|
|
|
|
if (v == lev) wf[2][p].usDelay += bitDelay[b+1];
|
|
else
|
|
{
|
|
p++;
|
|
|
|
lev = v;
|
|
|
|
if (lev)
|
|
{
|
|
wf[2][p].gpioOn = (1<<gpio);
|
|
wf[2][p].gpioOff = 0;
|
|
wf[2][p].flags = 0;
|
|
}
|
|
else
|
|
{
|
|
wf[2][p].gpioOn = 0;
|
|
wf[2][p].gpioOff = (1<<gpio);
|
|
wf[2][p].flags = 0;
|
|
}
|
|
|
|
wf[2][p].usDelay = bitDelay[b+1];
|
|
}
|
|
}
|
|
|
|
/* stop bit */
|
|
|
|
if (lev) wf[2][p].usDelay += bitDelay[data_bits+1];
|
|
else
|
|
{
|
|
p++;
|
|
|
|
wf[2][p].gpioOn = (1<<gpio);
|
|
wf[2][p].gpioOff = 0;
|
|
wf[2][p].usDelay = bitDelay[data_bits+1];
|
|
wf[2][p].flags = 0;
|
|
}
|
|
}
|
|
|
|
p++;
|
|
|
|
wf[2][p].gpioOn = (1<<gpio);
|
|
wf[2][p].gpioOff = 0;
|
|
wf[2][p].usDelay = bitDelay[0];
|
|
wf[2][p].flags = 0;
|
|
|
|
return rawWaveAddGeneric(p, wf[2]);
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int rawWaveAddSPI(
|
|
rawSPI_t *spi,
|
|
unsigned offset,
|
|
unsigned spiSS,
|
|
char *buf,
|
|
unsigned spiTxBits,
|
|
unsigned spiBitFirst,
|
|
unsigned spiBitLast,
|
|
unsigned spiBits)
|
|
{
|
|
int p, bit, dbv, halfbit;
|
|
int rising_edge[2], read_cycle[2];
|
|
uint32_t on_bits, off_bits;
|
|
int tx_bit_pos;
|
|
|
|
DBG(DBG_USER,
|
|
"spi=%08"PRIXPTR" off=%d spiSS=%d tx=%08"PRIXPTR", num=%d fb=%d lb=%d spiBits=%d",
|
|
(uintptr_t)spi, offset, spiSS, (uintptr_t)buf, spiTxBits,
|
|
spiBitFirst, spiBitLast, spiBits);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (spiSS > PI_MAX_USER_GPIO)
|
|
SOFT_ERROR(PI_BAD_USER_GPIO, "bad gpio (%d)", spiSS);
|
|
|
|
/*
|
|
CPOL CPHA
|
|
0 0 read rising/write falling
|
|
0 1 read falling/write rising
|
|
1 0 read falling/write rising
|
|
1 1 read rising/write falling
|
|
*/
|
|
|
|
if (spi->clk_pol) {rising_edge[0] = 0; rising_edge[1] = 1;}
|
|
else {rising_edge[0] = 1; rising_edge[1] = 0;}
|
|
|
|
if (spi->clk_pha) {read_cycle[0] = 0; read_cycle[1] = 1;}
|
|
else {read_cycle[0] = 1; read_cycle[1] = 0;}
|
|
|
|
p = 0;
|
|
|
|
if (offset)
|
|
{
|
|
wf[2][p].gpioOn = 0;
|
|
wf[2][p].gpioOff = 0;
|
|
wf[2][p].flags = 0;
|
|
wf[2][p].usDelay = offset;
|
|
p++;
|
|
}
|
|
|
|
on_bits = 0;
|
|
off_bits = 0;
|
|
|
|
tx_bit_pos = 0;
|
|
|
|
/* preset initial mosi bit */
|
|
|
|
if (getBitInBytes(tx_bit_pos, buf, spiTxBits))
|
|
{
|
|
on_bits |= (1<<(spi->mosi));
|
|
dbv = 1;
|
|
}
|
|
else
|
|
{
|
|
off_bits |= (1<<(spi->mosi));
|
|
dbv = 0;
|
|
}
|
|
|
|
if (!spi->clk_pha) tx_bit_pos ++;
|
|
|
|
if (spi->ss_pol) off_bits |= (1<<spiSS);
|
|
else on_bits |= (1<<spiSS);
|
|
|
|
if (spi->clk_pol) on_bits |= (1<<(spi->clk));
|
|
else off_bits |= (1<<(spi->clk));
|
|
|
|
wf[2][p].gpioOn = on_bits;
|
|
wf[2][p].gpioOff = off_bits;
|
|
wf[2][p].flags = 0;
|
|
|
|
if (spi->clk_us > spi->ss_us) wf[2][p].usDelay = spi->clk_us;
|
|
else wf[2][p].usDelay = spi->ss_us;
|
|
|
|
p++;
|
|
|
|
for (bit=1; bit<=spiBits; bit++)
|
|
{
|
|
for (halfbit=0; halfbit<2; halfbit++)
|
|
{
|
|
wf[2][p].usDelay = spi->clk_us;
|
|
wf[2][p].flags = 0;
|
|
|
|
on_bits = 0;
|
|
off_bits = 0;
|
|
|
|
if (read_cycle[halfbit])
|
|
{
|
|
if ((bit>=spiBitFirst) && (bit<=spiBitLast))
|
|
wf[2][p].flags = WAVE_FLAG_READ;
|
|
}
|
|
else
|
|
{
|
|
if (getBitInBytes(tx_bit_pos, buf, spiTxBits))
|
|
{
|
|
if (!dbv) on_bits |= (1<<(spi->mosi));
|
|
dbv = 1;
|
|
}
|
|
else
|
|
{
|
|
if (dbv) off_bits |= (1<<(spi->mosi));
|
|
dbv = 0;
|
|
}
|
|
|
|
++tx_bit_pos;
|
|
}
|
|
|
|
if (rising_edge[halfbit]) on_bits |= (1<<(spi->clk));
|
|
else off_bits |= (1<<(spi->clk));
|
|
|
|
wf[2][p].gpioOn = on_bits;
|
|
wf[2][p].gpioOff = off_bits;
|
|
|
|
p++;
|
|
}
|
|
}
|
|
|
|
on_bits = 0;
|
|
off_bits = 0;
|
|
|
|
if (spi->ss_pol) on_bits |= (1<<spiSS);
|
|
else off_bits |= (1<<spiSS);
|
|
|
|
wf[2][p].gpioOn = on_bits;
|
|
wf[2][p].gpioOff = off_bits;
|
|
wf[2][p].flags = 0;
|
|
wf[2][p].usDelay = 0;
|
|
|
|
p++;
|
|
|
|
return rawWaveAddGeneric(p, wf[2]);
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioWaveCreate(void)
|
|
{
|
|
int i, wid;
|
|
int numCB, numBOOL, numTOOL;
|
|
int CB, BOOL, TOOL;
|
|
|
|
DBG(DBG_USER, "");
|
|
|
|
CHECK_INITED;
|
|
|
|
if (wfc[wfcur] == 0) return PI_EMPTY_WAVEFORM;
|
|
|
|
/* What resources are needed? */
|
|
|
|
waveCBsOOLs(&numCB, &numBOOL, &numTOOL);
|
|
|
|
wid = -1;
|
|
|
|
/* Is there an exact fit with a deleted wave. */
|
|
|
|
for (i=0; i<waveOutCount; i++)
|
|
{
|
|
if (waveInfo[i].deleted &&
|
|
(waveInfo[i].numCB == numCB) &&
|
|
(waveInfo[i].numBOOL == numBOOL) &&
|
|
(waveInfo[i].numTOOL == numTOOL))
|
|
{
|
|
/* Reuse the deleted waves resources. */
|
|
wid = i;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (wid == -1)
|
|
{
|
|
/* Are there enough spare resources? */
|
|
|
|
if ((numCB+waveOutBotCB) > NUM_WAVE_CBS)
|
|
return PI_TOO_MANY_CBS;
|
|
|
|
if ((numBOOL+waveOutBotOOL) > (waveOutTopOOL-numTOOL))
|
|
return PI_TOO_MANY_OOL;
|
|
|
|
if (wid >= PI_MAX_WAVES)
|
|
return PI_NO_WAVEFORM_ID;
|
|
|
|
wid = waveOutCount++;
|
|
|
|
waveInfo[wid].botCB = waveOutBotCB;
|
|
waveInfo[wid].topCB = waveOutBotCB + numCB -1;
|
|
waveInfo[wid].botOOL = waveOutBotOOL;
|
|
waveInfo[wid].topOOL = waveOutTopOOL;
|
|
waveInfo[wid].numCB = numCB;
|
|
waveInfo[wid].numBOOL = numBOOL;
|
|
waveInfo[wid].numTOOL = numTOOL;
|
|
|
|
waveOutBotCB += numCB;
|
|
waveOutBotOOL += numBOOL;
|
|
waveOutTopOOL -= numTOOL;
|
|
}
|
|
|
|
/* Must be room if got this far. */
|
|
|
|
CB = waveInfo[wid].botCB;
|
|
BOOL = waveInfo[wid].botOOL;
|
|
TOOL = waveInfo[wid].topOOL;
|
|
|
|
wave2Cbs(PI_WAVE_MODE_ONE_SHOT, &CB, &BOOL, &TOOL, 0, 0, 0);
|
|
|
|
/* Sanity check. */
|
|
|
|
if ( (numCB != (CB-waveInfo[wid].botCB)) ||
|
|
(numBOOL != (BOOL-waveInfo[wid].botOOL)) ||
|
|
(numTOOL != (waveInfo[wid].topOOL-TOOL)) )
|
|
{
|
|
DBG(DBG_ALWAYS, "ERROR wid=%d CBs %d=%d BOOL %d=%d TOOL %d=%d", wid,
|
|
numCB, CB-waveInfo[wid].botCB,
|
|
numBOOL, BOOL-waveInfo[wid].botOOL,
|
|
numTOOL, waveInfo[wid].topOOL-TOOL);
|
|
}
|
|
|
|
DBG(DBG_USER, "Wave Stats: wid=%d CBs %d BOOL %d TOOL %d", wid,
|
|
numCB, numBOOL, numTOOL);
|
|
|
|
waveInfo[wid].deleted = 0;
|
|
|
|
/* Consume waves. */
|
|
|
|
wfc[0] = 0;
|
|
wfc[1] = 0;
|
|
wfc[2] = 0;
|
|
|
|
wfcur = 0;
|
|
|
|
return wid;
|
|
}
|
|
|
|
int gpioWaveCreatePad(int pctCB, int pctBOOL, int pctTOOL)
|
|
{
|
|
int i, wid;
|
|
int numCB, numBOOL, numTOOL;
|
|
int CB, BOOL, TOOL;
|
|
|
|
DBG(DBG_USER, "%d, %d, %d", pctCB, pctBOOL, pctTOOL);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (pctCB < 0 || pctCB > 100)
|
|
SOFT_ERROR(PI_BAD_PARAM, "bad wave param, pctCB=(%d)", pctCB);
|
|
if (pctBOOL < 0 || pctBOOL > 100)
|
|
SOFT_ERROR(PI_BAD_PARAM, "bad wave param, pctBOOL=(%d)", pctBOOL);
|
|
if (pctTOOL < 0 || pctTOOL > 100)
|
|
SOFT_ERROR(PI_BAD_PARAM, "bad wave param, pctTOOL=(%d)", pctTOOL);
|
|
|
|
if (wfc[wfcur] == 0) return PI_EMPTY_WAVEFORM;
|
|
|
|
/* What resources are needed? */
|
|
waveCBsOOLs(&numCB, &numBOOL, &numTOOL);
|
|
|
|
/* Amount of pad required */
|
|
CB = (NUM_WAVE_CBS - PI_WAVE_COUNT_PAGES*CBS_PER_OPAGE) * pctCB / 100;
|
|
BOOL = (NUM_WAVE_OOL - PI_WAVE_COUNT_PAGES*OOL_PER_OPAGE) * pctBOOL /100;
|
|
TOOL = (NUM_WAVE_OOL - PI_WAVE_COUNT_PAGES*OOL_PER_OPAGE) * pctTOOL /100;
|
|
|
|
/* Reject if wave is too big */
|
|
if (numCB > CB) return PI_TOO_MANY_CBS;
|
|
if (numBOOL > BOOL) return PI_TOO_MANY_OOL;
|
|
if (numTOOL > TOOL) return PI_TOO_MANY_OOL;
|
|
|
|
/* Set the padding */
|
|
numCB = CB;
|
|
numBOOL = BOOL;
|
|
numTOOL = TOOL;
|
|
|
|
|
|
wid = -1;
|
|
|
|
/* Is there an exact fit with a deleted wave. */
|
|
|
|
for (i=0; i<waveOutCount; i++)
|
|
{
|
|
if (waveInfo[i].deleted &&
|
|
(waveInfo[i].numCB == numCB) &&
|
|
(waveInfo[i].numBOOL == numBOOL) &&
|
|
(waveInfo[i].numTOOL == numTOOL))
|
|
{
|
|
/* Reuse the deleted waves resources. */
|
|
wid = i;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (wid == -1)
|
|
{
|
|
/* Are there enough spare resources? */
|
|
|
|
if ((numCB+waveOutBotCB) > NUM_WAVE_CBS)
|
|
return PI_TOO_MANY_CBS;
|
|
|
|
if ((numBOOL+waveOutBotOOL) > (waveOutTopOOL-numTOOL))
|
|
return PI_TOO_MANY_OOL;
|
|
|
|
if (wid >= PI_MAX_WAVES)
|
|
return PI_NO_WAVEFORM_ID;
|
|
|
|
wid = waveOutCount++;
|
|
|
|
waveInfo[wid].botCB = waveOutBotCB;
|
|
waveInfo[wid].topCB = waveOutBotCB + numCB -1;
|
|
waveInfo[wid].botOOL = waveOutBotOOL;
|
|
waveInfo[wid].topOOL = waveOutTopOOL;
|
|
waveInfo[wid].numCB = numCB;
|
|
waveInfo[wid].numBOOL = numBOOL;
|
|
waveInfo[wid].numTOOL = numTOOL;
|
|
|
|
waveOutBotCB += numCB;
|
|
waveOutBotOOL += numBOOL;
|
|
waveOutTopOOL -= numTOOL;
|
|
}
|
|
|
|
/* Must be room if got this far. */
|
|
|
|
CB = waveInfo[wid].botCB;
|
|
BOOL = waveInfo[wid].botOOL;
|
|
TOOL = waveInfo[wid].topOOL;
|
|
|
|
wave2Cbs(PI_WAVE_MODE_ONE_SHOT, &CB, &BOOL, &TOOL, numCB, numBOOL, numTOOL);
|
|
|
|
/* Sanity check. */
|
|
|
|
if ( (numCB != (CB-waveInfo[wid].botCB)) ||
|
|
(numBOOL != (BOOL-waveInfo[wid].botOOL)) ||
|
|
(numTOOL != (waveInfo[wid].topOOL-TOOL)) )
|
|
{
|
|
DBG(DBG_ALWAYS, "ERROR wid=%d CBs %d=%d BOOL %d=%d TOOL %d=%d", wid,
|
|
numCB, CB-waveInfo[wid].botCB,
|
|
numBOOL, BOOL-waveInfo[wid].botOOL,
|
|
numTOOL, waveInfo[wid].topOOL-TOOL);
|
|
}
|
|
|
|
DBG(DBG_USER, "Wave padding: wid=%d CBs %d BOOL %d TOOL %d", wid,
|
|
numCB, numBOOL, numTOOL);
|
|
|
|
waveInfo[wid].deleted = 0;
|
|
|
|
/* Consume waves. */
|
|
|
|
wfc[0] = 0;
|
|
wfc[1] = 0;
|
|
wfc[2] = 0;
|
|
|
|
wfcur = 0;
|
|
|
|
return wid;
|
|
}
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioWaveDelete(unsigned wave_id)
|
|
{
|
|
DBG(DBG_USER, "wave id=%d", wave_id);
|
|
|
|
CHECK_INITED;
|
|
|
|
if ((wave_id >= waveOutCount) || waveInfo[wave_id].deleted)
|
|
SOFT_ERROR(PI_BAD_WAVE_ID, "bad wave id (%d)", wave_id);
|
|
|
|
waveInfo[wave_id].deleted = 1;
|
|
|
|
if (wave_id == (waveOutCount-1))
|
|
{
|
|
/* top wave deleted, garbage collect any other deleted waves */
|
|
|
|
while ((wave_id > 0) && (waveInfo[wave_id-1].deleted)) --wave_id;
|
|
|
|
waveOutBotCB = waveInfo[wave_id].botCB;
|
|
waveOutBotOOL = waveInfo[wave_id].botOOL;
|
|
waveOutTopOOL = waveInfo[wave_id].topOOL;
|
|
|
|
waveOutCount = wave_id;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioWaveTxStart(unsigned wave_mode)
|
|
{
|
|
/* This function is deprecated and has been removed. */
|
|
|
|
CHECK_INITED;
|
|
|
|
SOFT_ERROR(PI_DEPRECATED, "deprected function removed");
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioWaveTxSend(unsigned wave_id, unsigned wave_mode)
|
|
{
|
|
rawCbs_t *p=NULL;
|
|
|
|
DBG(DBG_USER, "wave_id=%d wave_mode=%d", wave_id, wave_mode);
|
|
|
|
CHECK_INITED;
|
|
|
|
if ((wave_id >= waveOutCount) || waveInfo[wave_id].deleted)
|
|
SOFT_ERROR(PI_BAD_WAVE_ID, "bad wave id (%d)", wave_id);
|
|
|
|
if (wave_mode > PI_WAVE_MODE_REPEAT_SYNC)
|
|
SOFT_ERROR(PI_BAD_WAVE_MODE, "bad wave mode (%d)", wave_mode);
|
|
|
|
if (!waveClockInited)
|
|
{
|
|
stopHardwarePWM();
|
|
initClock(0); /* initialise secondary clock */
|
|
waveClockInited = 1;
|
|
PWMClockInited = 0;
|
|
}
|
|
|
|
if (wave_mode < PI_WAVE_MODE_ONE_SHOT_SYNC) initKillDMA(dmaOut);
|
|
|
|
p = rawWaveCBAdr(waveInfo[wave_id].topCB);
|
|
|
|
if ((wave_mode & 1) == PI_WAVE_MODE_ONE_SHOT)
|
|
p->next = 0;
|
|
else
|
|
p->next = waveCbPOadr(waveInfo[wave_id].botCB+1);
|
|
|
|
if (waveEndPtr && (wave_mode > PI_WAVE_MODE_REPEAT))
|
|
{
|
|
*waveEndPtr = waveCbPOadr(waveInfo[wave_id].botCB+1);
|
|
|
|
if (!dmaOut[DMA_CONBLK_AD])
|
|
{
|
|
initDMAgo((uint32_t *)dmaOut, waveCbPOadr(waveInfo[wave_id].botCB));
|
|
}
|
|
}
|
|
else
|
|
{
|
|
initDMAgo((uint32_t *)dmaOut, waveCbPOadr(waveInfo[wave_id].botCB));
|
|
}
|
|
|
|
waveEndPtr = &p->next;
|
|
|
|
/* for compatability with the deprecated gpioWaveTxStart return the
|
|
number of cbs
|
|
*/
|
|
return (waveInfo[wave_id].topCB - waveInfo[wave_id].botCB) + 1;
|
|
}
|
|
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static int chainGetCB(int n)
|
|
{
|
|
int block, index;
|
|
|
|
if (n < (WCB_CHAIN_CBS * PI_WAVE_COUNT_PAGES))
|
|
{
|
|
block = n / WCB_CHAIN_CBS;
|
|
index = n % WCB_CHAIN_CBS;
|
|
return (block*CBS_PER_OPAGE) + WCB_COUNTER_CBS + index;
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
static void chainSetVal(int n, uint32_t val)
|
|
{
|
|
int block, index;
|
|
uint32_t *p;
|
|
|
|
if (n < (WCB_CHAIN_OOL * PI_WAVE_COUNT_PAGES))
|
|
{
|
|
block = n / WCB_CHAIN_OOL;
|
|
index = n % WCB_CHAIN_OOL;
|
|
p = (uint32_t *) dmaOVirt[block] + (WCB_COUNTER_CBS+WCB_CHAIN_CBS) * 8;
|
|
p[index] = val;
|
|
}
|
|
}
|
|
|
|
static uint32_t chainGetValPadr(int n)
|
|
{
|
|
int block, index;
|
|
uint32_t *p;
|
|
|
|
if (n < (WCB_CHAIN_OOL * PI_WAVE_COUNT_PAGES))
|
|
{
|
|
block = n / WCB_CHAIN_OOL;
|
|
index = n % WCB_CHAIN_OOL;
|
|
p = (uint32_t *) dmaOBus[block] + (WCB_COUNTER_CBS+WCB_CHAIN_CBS) * 8;
|
|
//cast twice to suppress warning, I belive this is ok as dmaOBus
|
|
//contains bus addresses not virtual addresses.
|
|
return (uint32_t)(uintptr_t) (p + index);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static uint32_t chainGetCntVal(int counter, int slot)
|
|
{
|
|
uint32_t *p;
|
|
int page, offset;
|
|
page = counter / 2;
|
|
offset = (counter % 2 ? WCB_COUNTER_OOL : 0);
|
|
p = (uint32_t *) dmaOVirt[page] + (WCB_COUNTER_CBS+WCB_CHAIN_CBS) * 8;
|
|
return p[WCB_CHAIN_OOL+ offset + slot];
|
|
}
|
|
|
|
static void chainSetCntVal(int counter, int slot, uint32_t value)
|
|
{
|
|
uint32_t *p;
|
|
int page, offset;
|
|
page = counter / 2;
|
|
offset = (counter % 2 ? WCB_COUNTER_OOL : 0);
|
|
p = (uint32_t *) dmaOVirt[page] + (WCB_COUNTER_CBS+WCB_CHAIN_CBS) * 8;
|
|
p[WCB_CHAIN_OOL + offset + slot] = value;
|
|
}
|
|
|
|
static uint32_t chainGetCntValPadr(int counter, int slot)
|
|
{
|
|
uint32_t *p;
|
|
int page, offset;
|
|
page = counter / 2;
|
|
offset = (counter % 2 ? WCB_COUNTER_OOL : 0);
|
|
p = (uint32_t *) dmaOBus[page] + (WCB_COUNTER_CBS+WCB_CHAIN_CBS) * 8;
|
|
//cast twice to suppress warning, I belive this is ok as dmaOBus
|
|
//contains bus addresses not virtual addresses. --plugwash
|
|
return (uint32_t)(uintptr_t)(p + WCB_CHAIN_OOL + offset + slot);
|
|
}
|
|
|
|
static int chainGetCntCB(int counter)
|
|
{
|
|
int page, offset;
|
|
page = counter / 2;
|
|
offset = (counter % 2 ? WCB_CNT_CBS : 0);
|
|
return ((page * CBS_PER_OPAGE) + offset);
|
|
}
|
|
|
|
static void chainMakeCounter(
|
|
unsigned counter,
|
|
unsigned blklen,
|
|
unsigned blocks,
|
|
unsigned count,
|
|
uint32_t repeat,
|
|
uint32_t next)
|
|
{
|
|
rawCbs_t *p=NULL;
|
|
|
|
int b, baseCB, dig;
|
|
uint32_t nxt;
|
|
|
|
int botCB;
|
|
|
|
botCB = chainGetCntCB(counter);
|
|
|
|
baseCB = botCB;
|
|
|
|
/* set up all the OOLs */
|
|
for (b=0; b < (blocks*(blklen+1)); b++) chainSetCntVal(counter, b, repeat);
|
|
|
|
for (b=0; b<blocks; b++)
|
|
chainSetCntVal(counter,
|
|
((b*(blklen+1))+blklen),
|
|
waveCbPOadr(baseCB+((b*3)+3)));
|
|
|
|
for (b=0; b<blocks; b++)
|
|
{
|
|
/* copy BOTTOM to NEXT */
|
|
|
|
p = rawWaveCBAdr(botCB++);
|
|
|
|
p->info = NORMAL_DMA;
|
|
|
|
p->src = chainGetCntValPadr(counter, b*(blklen+1));
|
|
p->dst = (waveCbPOadr(botCB+1) + 20);
|
|
|
|
p->length = 4;
|
|
p->next = waveCbPOadr(botCB);
|
|
|
|
/* copy BOTTOM to TOP */
|
|
|
|
p = rawWaveCBAdr(botCB++);
|
|
|
|
p->info = NORMAL_DMA;
|
|
|
|
p->src = chainGetCntValPadr(counter, b*(blklen+1));
|
|
p->dst = chainGetCntValPadr(counter, (b*(blklen+1))+blklen);
|
|
|
|
p->length = 4;
|
|
p->next = waveCbPOadr(botCB);
|
|
|
|
/* shift all down one */
|
|
|
|
p = rawWaveCBAdr(botCB++);
|
|
|
|
p->info = NORMAL_DMA|DMA_SRC_INC|DMA_DEST_INC;
|
|
|
|
p->src = chainGetCntValPadr(counter, ((b*(blklen+1))+1));
|
|
p->dst = chainGetCntValPadr(counter, ((b*(blklen+1))+0));
|
|
|
|
p->length = blklen*4;
|
|
p->next = repeat;
|
|
}
|
|
|
|
/* reset the counter */
|
|
|
|
p = rawWaveCBAdr(botCB);
|
|
|
|
p->info = NORMAL_DMA|DMA_SRC_INC|DMA_DEST_INC;
|
|
|
|
p->src = chainGetCntValPadr(counter, blocks*(blklen+1));
|
|
p->dst = chainGetCntValPadr(counter, 0);
|
|
|
|
p->length = blocks*(blklen+1)*4;
|
|
p->next = next;
|
|
|
|
b = 0;
|
|
|
|
while (count && (b<blocks))
|
|
{
|
|
dig = count % blklen;
|
|
count /= blklen;
|
|
|
|
if (count) nxt = chainGetCntVal(counter, (b*(blklen+1))+blklen);
|
|
else nxt = waveCbPOadr(botCB);
|
|
|
|
chainSetCntVal(counter, b*(blklen+1)+dig, nxt);
|
|
|
|
b++;
|
|
}
|
|
|
|
/* copy all the OOLs */
|
|
for (b=0; b < (blocks*(blklen+1)); b++)
|
|
chainSetCntVal(
|
|
counter, b+(blocks*(blklen+1)), chainGetCntVal(counter, b));
|
|
}
|
|
|
|
|
|
int gpioWaveChain(char *buf, unsigned bufSize)
|
|
{
|
|
unsigned blklen=16, blocks=4;
|
|
int cb, chaincb;
|
|
rawCbs_t *p;
|
|
int i, wid, cmd, loop, counters;
|
|
unsigned cycles, delayCBs, dcb, delayLeft;
|
|
uint32_t repeat, next, *endPtr;
|
|
int stk_pos[10], stk_lev=0;
|
|
|
|
cb = 0;
|
|
loop = -1;
|
|
|
|
DBG(DBG_USER, "bufSize=%d [%s]", bufSize, myBuf2Str(bufSize, buf));
|
|
|
|
CHECK_INITED;
|
|
|
|
if (!waveClockInited)
|
|
{
|
|
stopHardwarePWM();
|
|
initClock(0); /* initialise secondary clock */
|
|
waveClockInited = 1;
|
|
PWMClockInited = 0;
|
|
}
|
|
|
|
initKillDMA(dmaOut);
|
|
|
|
waveEndPtr = NULL;
|
|
endPtr = NULL;
|
|
|
|
/* add delay cb at start of DMA */
|
|
|
|
p = rawWaveCBAdr(chainGetCB(cb++));
|
|
|
|
/* use the secondary clock */
|
|
|
|
if (gpioCfg.clockPeriph != PI_CLOCK_PCM)
|
|
{
|
|
p->info = NORMAL_DMA | TIMED_DMA(2);
|
|
p->dst = PCM_TIMER;
|
|
}
|
|
else
|
|
{
|
|
p->info = NORMAL_DMA | TIMED_DMA(5);
|
|
p->dst = PWM_TIMER;
|
|
}
|
|
|
|
//cast twice to suppress warning, I belive this is ok as dmaOBus
|
|
//contains bus addresses not virtual addresses. --plugwash
|
|
p->src = (uint32_t)(uintptr_t) (&dmaOBus[0]->periphData);
|
|
p->length = BPD * 20 / PI_WF_MICROS; /* 20 micros delay */
|
|
p->next = waveCbPOadr(chainGetCB(cb));
|
|
|
|
counters = 0;
|
|
wid = -1;
|
|
|
|
i = 0;
|
|
|
|
while (i<bufSize)
|
|
{
|
|
wid = (unsigned)buf[i];
|
|
|
|
if (wid == 255) /* wave command */
|
|
{
|
|
if ((i+2) > bufSize)
|
|
SOFT_ERROR(PI_BAD_CHAIN_CMD,
|
|
"incomplete chain command (at %d)", i);
|
|
|
|
cmd = buf[i+1];
|
|
|
|
if (cmd == 0) /* loop begin */
|
|
{
|
|
if (stk_lev >= (sizeof(stk_pos)/sizeof(int)))
|
|
SOFT_ERROR(PI_CHAIN_NESTING,
|
|
"chain counters nested too deep (at %d)", i);
|
|
|
|
stk_pos[stk_lev++] = cb;
|
|
|
|
i += 2;
|
|
}
|
|
else if (cmd == 1) /* loop end */
|
|
{
|
|
if (counters >= WCB_COUNTERS)
|
|
SOFT_ERROR(PI_CHAIN_COUNTER,
|
|
"too many chain counters (at %d)", i);
|
|
|
|
if ((i+4) > bufSize)
|
|
SOFT_ERROR(PI_BAD_CHAIN_CMD,
|
|
"incomplete chain command (at %d)", i);
|
|
|
|
loop = 0;
|
|
if (--stk_lev >= 0) loop = stk_pos[stk_lev];
|
|
|
|
if ((loop < 1) || (loop == cb))
|
|
SOFT_ERROR(PI_BAD_CHAIN_LOOP,
|
|
"empty chain loop (at %d)", i);
|
|
|
|
cycles = ((unsigned)buf[i+3] << 8) + (unsigned)buf[i+2];
|
|
|
|
i += 4;
|
|
|
|
if (cycles > PI_MAX_WAVE_CYCLES)
|
|
SOFT_ERROR(PI_CHAIN_LOOP_CNT,
|
|
"bad chain loop count (%d)", cycles);
|
|
|
|
if (cycles == 0)
|
|
{
|
|
/* Skip the complete loop block. Change
|
|
the next pointing to the start of the
|
|
loop block to the current cb.
|
|
*/
|
|
p = rawWaveCBAdr(chainGetCB(loop));
|
|
p->next = waveCbPOadr(chainGetCB(cb));
|
|
}
|
|
else if (cycles == 1)
|
|
{
|
|
/* Nothing to do, no need for a counter. */
|
|
}
|
|
else
|
|
{
|
|
chaincb = chainGetCB(cb++);
|
|
if (chaincb < 0)
|
|
SOFT_ERROR(PI_CHAIN_TOO_BIG, "chain is too long (%d)", cb);
|
|
|
|
p = rawWaveCBAdr(chaincb);
|
|
|
|
repeat = waveCbPOadr(chainGetCB(loop));
|
|
|
|
/* Need to check next cb as well. */
|
|
|
|
chaincb = chainGetCB(cb);
|
|
|
|
if (chaincb < 0)
|
|
SOFT_ERROR(PI_CHAIN_TOO_BIG, "chain is too long (%d)", cb);
|
|
|
|
next = waveCbPOadr(chainGetCB(cb));
|
|
|
|
/* dummy src and dest */
|
|
p->info = NORMAL_DMA;
|
|
//cast twice to suppress warning, I belive this is ok as dmaOBus
|
|
//contains bus addresses not virtual addresses. --plugwash
|
|
p->src = (uint32_t)(uintptr_t) (&dmaOBus[0]->periphData);
|
|
p->dst = (uint32_t)(uintptr_t) (&dmaOBus[0]->periphData);
|
|
p->length = 4;
|
|
p->next = waveCbPOadr(chainGetCntCB(counters));
|
|
|
|
chainMakeCounter(counters, blklen, blocks,
|
|
cycles-1, repeat, next);
|
|
|
|
counters++;
|
|
}
|
|
}
|
|
else if (cmd == 2) /* delay us */
|
|
{
|
|
if ((i+4) > bufSize)
|
|
SOFT_ERROR(PI_BAD_CHAIN_CMD,
|
|
"incomplete chain command (at %d)", i);
|
|
|
|
cycles = ((unsigned)buf[i+3] << 8) + (unsigned)buf[i+2];
|
|
|
|
i += 4;
|
|
|
|
if (cycles > PI_MAX_WAVE_DELAY)
|
|
SOFT_ERROR(PI_BAD_CHAIN_DELAY,
|
|
"bad chain delay micros (%d)", cycles);
|
|
|
|
if (cycles)
|
|
{
|
|
delayLeft = cycles;
|
|
delayCBs = waveDelayCBs(delayLeft);
|
|
for (dcb=0; dcb<delayCBs; dcb++)
|
|
{
|
|
chaincb = chainGetCB(cb++);
|
|
|
|
if (chaincb < 0)
|
|
SOFT_ERROR(
|
|
PI_CHAIN_TOO_BIG, "chain is too long (%d)", cb);
|
|
|
|
p = rawWaveCBAdr(chaincb);
|
|
|
|
/* use the secondary clock */
|
|
|
|
if (gpioCfg.clockPeriph != PI_CLOCK_PCM)
|
|
{
|
|
p->info = NORMAL_DMA | TIMED_DMA(2);
|
|
p->dst = PCM_TIMER;
|
|
}
|
|
else
|
|
{
|
|
p->info = NORMAL_DMA | TIMED_DMA(5);
|
|
p->dst = PWM_TIMER;
|
|
}
|
|
|
|
//cast twice to suppress warning, I belive this is ok as dmaOBus
|
|
//contains bus addresses not virtual addresses. --plugwash
|
|
p->src = (uint32_t)(uintptr_t) (&dmaOBus[0]->periphData);
|
|
|
|
p->length = BPD * delayLeft / PI_WF_MICROS;
|
|
|
|
if ((gpioCfg.DMAsecondaryChannel >= DMA_LITE_FIRST) &&
|
|
(p->length > DMA_LITE_MAX))
|
|
{
|
|
p->length = DMA_LITE_MAX;
|
|
}
|
|
|
|
delayLeft -= (p->length / BPD);
|
|
|
|
p->next = waveCbPOadr(chainGetCB(cb));
|
|
}
|
|
}
|
|
}
|
|
else if (cmd == 3) /* repeat loop forever */
|
|
{
|
|
i += 2;
|
|
|
|
loop = 0;
|
|
if (--stk_lev >= 0) loop = stk_pos[stk_lev];
|
|
|
|
if ((loop < 1) || (loop == cb))
|
|
SOFT_ERROR(PI_BAD_CHAIN_LOOP,
|
|
"empty chain loop (at %d)", i);
|
|
|
|
chaincb = chainGetCB(cb++);
|
|
if (chaincb < 0)
|
|
SOFT_ERROR(PI_CHAIN_TOO_BIG, "chain is too long (%d)", cb);
|
|
|
|
if (i < bufSize)
|
|
SOFT_ERROR(PI_BAD_FOREVER,
|
|
"loop forever must be last command");
|
|
|
|
p = rawWaveCBAdr(chaincb);
|
|
|
|
/* dummy src and dest */
|
|
p->info = NORMAL_DMA;
|
|
//cast twice to suppress warning, I belive this is ok as dmaOBus
|
|
//contains bus addresses not virtual addresses. --plugwash
|
|
p->src = (uint32_t)(uintptr_t) (&dmaOBus[0]->periphData);
|
|
p->dst = (uint32_t)(uintptr_t) (&dmaOBus[0]->periphData);
|
|
p->length = 4;
|
|
p->next = waveCbPOadr(chainGetCB(loop));
|
|
endPtr = &p->next;
|
|
}
|
|
else
|
|
SOFT_ERROR(PI_BAD_CHAIN_CMD,
|
|
"unknown chain command (255 %d)", cmd);
|
|
}
|
|
else if ((wid >= waveOutCount) || waveInfo[wid].deleted)
|
|
SOFT_ERROR(PI_BAD_WAVE_ID, "undefined wave (%d)", wid);
|
|
else
|
|
{
|
|
chaincb = chainGetCB(cb++);
|
|
|
|
if (chaincb < 0)
|
|
SOFT_ERROR(PI_CHAIN_TOO_BIG, "chain is too long (%d)", cb);
|
|
|
|
p = rawWaveCBAdr(chaincb);
|
|
|
|
chaincb = chainGetCB(cb);
|
|
|
|
if (chaincb < 0)
|
|
SOFT_ERROR(PI_CHAIN_TOO_BIG, "chain is too long (%d)", cb);
|
|
|
|
chainSetVal(cb-1, waveCbPOadr(chaincb));
|
|
|
|
/* patch next of wid topCB to next cb */
|
|
|
|
p->info = NORMAL_DMA;
|
|
p->src = chainGetValPadr(cb-1); /* this next */
|
|
p->dst = waveCbPOadr(waveInfo[wid].topCB) + 20; /* wid next */
|
|
p->length = 4;
|
|
p->next = waveCbPOadr(waveInfo[wid].botCB+1);
|
|
|
|
i += 1;
|
|
}
|
|
}
|
|
|
|
chaincb = chainGetCB(cb++);
|
|
|
|
if (chaincb < 0)
|
|
SOFT_ERROR(PI_CHAIN_TOO_BIG, "chain is too long (%d)", cb);
|
|
|
|
p = rawWaveCBAdr(chaincb);
|
|
|
|
p->info = NORMAL_DMA;
|
|
|
|
//cast twice to suppress warning, I belive this is ok as dmaOBus
|
|
//contains bus addresses not virtual addresses. --plugwash
|
|
p->src = (uint32_t)(uintptr_t) (&dmaOBus[0]->periphData);
|
|
p->dst = (uint32_t)(uintptr_t) (&dmaOBus[0]->periphData);
|
|
p->length = 4;
|
|
p->next = 0;
|
|
|
|
if (!endPtr) endPtr = &p->next;
|
|
|
|
initDMAgo((uint32_t *)dmaOut, waveCbPOadr(chainGetCB(0)));
|
|
|
|
waveEndPtr = endPtr;
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*-------------------------------------------------------------------------*/
|
|
|
|
int gpioWaveTxBusy(void)
|
|
{
|
|
DBG(DBG_USER, "");
|
|
|
|
CHECK_INITED;
|
|
|
|
if (dmaOut[DMA_CONBLK_AD])
|
|
return 1;
|
|
else
|
|
return 0;
|
|
}
|
|
|
|
/*-------------------------------------------------------------------------*/
|
|
|
|
int gpioWaveTxAt(void)
|
|
{
|
|
int i, cb;
|
|
|
|
DBG(DBG_USER, "");
|
|
|
|
CHECK_INITED;
|
|
|
|
cb = dmaNowAtOCB();
|
|
|
|
if (cb < 0) return -cb;
|
|
|
|
for (i=0; i<PI_MAX_WAVES; i++)
|
|
{
|
|
if ( !waveInfo[i].deleted &&
|
|
(cb >= waveInfo[i].botCB) &&
|
|
(cb <= waveInfo[i].topCB) ) return i;
|
|
}
|
|
|
|
return PI_WAVE_NOT_FOUND;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioWaveTxStop(void)
|
|
{
|
|
DBG(DBG_USER, "");
|
|
|
|
CHECK_INITED;
|
|
|
|
initKillDMA(dmaOut);
|
|
|
|
waveEndPtr = NULL;
|
|
|
|
return 0;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioWaveGetMicros(void)
|
|
{
|
|
DBG(DBG_USER, "");
|
|
|
|
CHECK_INITED;
|
|
|
|
return wfStats.micros;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioWaveGetHighMicros(void)
|
|
{
|
|
DBG(DBG_USER, "");
|
|
|
|
CHECK_INITED;
|
|
|
|
return wfStats.highMicros;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioWaveGetMaxMicros(void)
|
|
{
|
|
DBG(DBG_USER, "");
|
|
|
|
CHECK_INITED;
|
|
|
|
return wfStats.maxMicros;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioWaveGetPulses(void)
|
|
{
|
|
DBG(DBG_USER, "");
|
|
|
|
CHECK_INITED;
|
|
|
|
return wfStats.pulses;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioWaveGetHighPulses(void)
|
|
{
|
|
DBG(DBG_USER, "");
|
|
|
|
CHECK_INITED;
|
|
|
|
return wfStats.highPulses;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioWaveGetMaxPulses(void)
|
|
{
|
|
DBG(DBG_USER, "");
|
|
|
|
CHECK_INITED;
|
|
|
|
return wfStats.maxPulses;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioWaveGetCbs(void)
|
|
{
|
|
DBG(DBG_USER, "");
|
|
|
|
CHECK_INITED;
|
|
|
|
return wfStats.cbs;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioWaveGetHighCbs(void)
|
|
{
|
|
DBG(DBG_USER, "");
|
|
|
|
CHECK_INITED;
|
|
|
|
return wfStats.highCbs;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioWaveGetMaxCbs(void)
|
|
{
|
|
DBG(DBG_USER, "");
|
|
|
|
CHECK_INITED;
|
|
|
|
return wfStats.maxCbs;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static int read_SDA(wfRx_t *w)
|
|
{
|
|
myGpioSetMode(w->I.SDA, PI_INPUT);
|
|
return myGpioRead(w->I.SDA);
|
|
}
|
|
|
|
static void set_SDA(wfRx_t *w)
|
|
{
|
|
myGpioSetMode(w->I.SDA, PI_INPUT);
|
|
}
|
|
|
|
static void clear_SDA(wfRx_t *w)
|
|
{
|
|
myGpioSetMode(w->I.SDA, PI_OUTPUT);
|
|
myGpioWrite(w->I.SDA, 0);
|
|
}
|
|
|
|
static void clear_SCL(wfRx_t *w)
|
|
{
|
|
myGpioSetMode(w->I.SCL, PI_OUTPUT);
|
|
myGpioWrite(w->I.SCL, 0);
|
|
}
|
|
|
|
static void I2C_delay(wfRx_t *w)
|
|
{
|
|
myGpioDelay(w->I.delay);
|
|
}
|
|
|
|
static void I2C_clock_stretch(wfRx_t *w)
|
|
{
|
|
uint32_t now, max_stretch=100000;
|
|
|
|
myGpioSetMode(w->I.SCL, PI_INPUT);
|
|
now = gpioTick();
|
|
while ((myGpioRead(w->I.SCL) == 0) && ((gpioTick()-now) < max_stretch));
|
|
}
|
|
|
|
static void I2CStart(wfRx_t *w)
|
|
{
|
|
if (w->I.started)
|
|
{
|
|
set_SDA(w);
|
|
I2C_delay(w);
|
|
I2C_clock_stretch(w);
|
|
I2C_delay(w);
|
|
}
|
|
|
|
clear_SDA(w);
|
|
I2C_delay(w);
|
|
clear_SCL(w);
|
|
I2C_delay(w);
|
|
|
|
w->I.started = 1;
|
|
}
|
|
|
|
static void I2CStop(wfRx_t *w)
|
|
{
|
|
clear_SDA(w);
|
|
I2C_delay(w);
|
|
I2C_clock_stretch(w);
|
|
I2C_delay(w);
|
|
set_SDA(w);
|
|
I2C_delay(w);
|
|
|
|
w->I.started = 0;
|
|
}
|
|
|
|
static void I2CPutBit(wfRx_t *w, int bit)
|
|
{
|
|
if (bit) set_SDA(w);
|
|
else clear_SDA(w);
|
|
|
|
I2C_delay(w);
|
|
I2C_clock_stretch(w);
|
|
I2C_delay(w);
|
|
clear_SCL(w);
|
|
}
|
|
|
|
static int I2CGetBit(wfRx_t *w)
|
|
{
|
|
int bit;
|
|
|
|
set_SDA(w); /* let SDA float */
|
|
I2C_delay(w);
|
|
I2C_clock_stretch(w);
|
|
bit = read_SDA(w);
|
|
I2C_delay(w);
|
|
clear_SCL(w);
|
|
|
|
return bit;
|
|
}
|
|
|
|
static int I2CPutByte(wfRx_t *w, int byte)
|
|
{
|
|
int bit, nack;
|
|
|
|
for(bit=0; bit<8; bit++)
|
|
{
|
|
I2CPutBit(w, byte & 0x80);
|
|
byte <<= 1;
|
|
}
|
|
|
|
nack = I2CGetBit(w);
|
|
|
|
return nack;
|
|
}
|
|
|
|
static uint8_t I2CGetByte(wfRx_t *w, int nack)
|
|
{
|
|
int bit, byte=0;
|
|
|
|
for (bit=0; bit<8; bit++)
|
|
{
|
|
byte = (byte << 1) | I2CGetBit(w);
|
|
}
|
|
|
|
I2CPutBit(w, nack);
|
|
|
|
return byte;
|
|
}
|
|
|
|
/*-------------------------------------------------------------------------*/
|
|
|
|
int bbI2COpen(unsigned SDA, unsigned SCL, unsigned baud)
|
|
{
|
|
DBG(DBG_USER, "SDA=%d SCL=%d baud=%d", SDA, SCL, baud);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (SDA > PI_MAX_USER_GPIO)
|
|
SOFT_ERROR(PI_BAD_USER_GPIO, "bad SDA (%d)", SDA);
|
|
|
|
if (SCL > PI_MAX_USER_GPIO)
|
|
SOFT_ERROR(PI_BAD_USER_GPIO, "bad SCL (%d)", SCL);
|
|
|
|
if ((baud < PI_BB_I2C_MIN_BAUD) || (baud > PI_BB_I2C_MAX_BAUD))
|
|
SOFT_ERROR(PI_BAD_I2C_BAUD,
|
|
"SDA %d, bad baud rate (%d)", SDA, baud);
|
|
|
|
if (wfRx[SDA].mode != PI_WFRX_NONE)
|
|
SOFT_ERROR(PI_GPIO_IN_USE, "gpio %d is already being used", SDA);
|
|
|
|
if ((wfRx[SCL].mode != PI_WFRX_NONE) || (SCL == SDA))
|
|
SOFT_ERROR(PI_GPIO_IN_USE, "gpio %d is already being used", SCL);
|
|
|
|
wfRx[SDA].gpio = SDA;
|
|
wfRx[SDA].mode = PI_WFRX_I2C_SDA;
|
|
wfRx[SDA].baud = baud;
|
|
|
|
wfRx[SDA].I.started = 0;
|
|
wfRx[SDA].I.SDA = SDA;
|
|
wfRx[SDA].I.SCL = SCL;
|
|
wfRx[SDA].I.delay = 500000 / baud;
|
|
wfRx[SDA].I.SDAMode = gpioGetMode(SDA);
|
|
wfRx[SDA].I.SCLMode = gpioGetMode(SCL);
|
|
|
|
wfRx[SCL].gpio = SCL;
|
|
wfRx[SCL].mode = PI_WFRX_I2C_SCL;
|
|
|
|
myGpioSetMode(SDA, PI_INPUT);
|
|
myGpioSetMode(SCL, PI_INPUT);
|
|
|
|
return 0;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int bbI2CClose(unsigned SDA)
|
|
{
|
|
DBG(DBG_USER, "SDA=%d", SDA);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (SDA > PI_MAX_USER_GPIO)
|
|
SOFT_ERROR(PI_BAD_USER_GPIO, "bad gpio (%d)", SDA);
|
|
|
|
switch(wfRx[SDA].mode)
|
|
{
|
|
case PI_WFRX_I2C_SDA:
|
|
|
|
myGpioSetMode(wfRx[SDA].I.SDA, wfRx[SDA].I.SDAMode);
|
|
myGpioSetMode(wfRx[SDA].I.SCL, wfRx[SDA].I.SCLMode);
|
|
|
|
wfRx[wfRx[SDA].I.SDA].mode = PI_WFRX_NONE;
|
|
wfRx[wfRx[SDA].I.SCL].mode = PI_WFRX_NONE;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
SOFT_ERROR(PI_NOT_I2C_GPIO, "no I2C on gpio (%d)", SDA);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*-------------------------------------------------------------------------*/
|
|
|
|
int bbI2CZip(
|
|
unsigned SDA,
|
|
char *inBuf,
|
|
unsigned inLen,
|
|
char *outBuf,
|
|
unsigned outLen)
|
|
{
|
|
int i, ack, inPos, outPos, status, bytes;
|
|
int addr, flags, esc, setesc;
|
|
wfRx_t *w;
|
|
|
|
DBG(DBG_USER, "gpio=%d inBuf=%s outBuf=%08"PRIXPTR" len=%d",
|
|
SDA, myBuf2Str(inLen, (char *)inBuf), (uintptr_t)outBuf, outLen);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (SDA > PI_MAX_USER_GPIO)
|
|
SOFT_ERROR(PI_BAD_USER_GPIO, "bad gpio (%d)", SDA);
|
|
|
|
if (wfRx[SDA].mode != PI_WFRX_I2C_SDA)
|
|
SOFT_ERROR(PI_NOT_I2C_GPIO, "no I2C on gpio (%d)", SDA);
|
|
|
|
if (!inBuf || !inLen)
|
|
SOFT_ERROR(PI_BAD_POINTER, "input buffer can't be NULL");
|
|
|
|
if (!outBuf && outLen)
|
|
SOFT_ERROR(PI_BAD_POINTER, "output buffer can't be NULL");
|
|
|
|
w = &wfRx[SDA];
|
|
|
|
inPos = 0;
|
|
outPos = 0;
|
|
status = 0;
|
|
|
|
addr = 0;
|
|
flags = 0;
|
|
esc = 0;
|
|
setesc = 0;
|
|
|
|
wfRx_lock(SDA);
|
|
|
|
while (!status && (inPos < inLen))
|
|
{
|
|
DBG(DBG_INTERNAL, "status=%d inpos=%d inlen=%d cmd=%d addr=%d flags=%x",
|
|
status, inPos, inLen, inBuf[inPos], addr, flags);
|
|
|
|
switch (inBuf[inPos++])
|
|
{
|
|
case PI_I2C_END:
|
|
status = 1;
|
|
break;
|
|
|
|
case PI_I2C_START:
|
|
I2CStart(w);
|
|
break;
|
|
|
|
case PI_I2C_STOP:
|
|
I2CStop(w);
|
|
break;
|
|
|
|
case PI_I2C_ADDR:
|
|
addr = myI2CGetPar(inBuf, &inPos, inLen, &esc);
|
|
if (addr < 0) status = PI_BAD_I2C_CMD;
|
|
break;
|
|
|
|
case PI_I2C_FLAGS:
|
|
/* cheat to force two byte flags */
|
|
esc = 1;
|
|
flags = myI2CGetPar(inBuf, &inPos, inLen, &esc);
|
|
if (flags < 0) status = PI_BAD_I2C_CMD;
|
|
break;
|
|
|
|
case PI_I2C_ESC:
|
|
setesc = 1;
|
|
break;
|
|
|
|
case PI_I2C_READ:
|
|
|
|
bytes = myI2CGetPar(inBuf, &inPos, inLen, &esc);
|
|
|
|
if (bytes >= 0) ack = I2CPutByte(w, (addr<<1)|1);
|
|
|
|
if (bytes > 0)
|
|
{
|
|
if (!ack)
|
|
{
|
|
if ((bytes + outPos) <= outLen)
|
|
{
|
|
for (i=0; i<(bytes-1); i++)
|
|
{
|
|
outBuf[outPos++] = I2CGetByte(w, 0);
|
|
}
|
|
outBuf[outPos++] = I2CGetByte(w, 1);
|
|
}
|
|
else status = PI_BAD_I2C_RLEN;
|
|
}
|
|
else status = PI_I2C_READ_FAILED;
|
|
}
|
|
else status = PI_BAD_I2C_CMD;
|
|
break;
|
|
|
|
case PI_I2C_WRITE:
|
|
|
|
bytes = myI2CGetPar(inBuf, &inPos, inLen, &esc);
|
|
|
|
if (bytes >= 0) ack = I2CPutByte(w, addr<<1);
|
|
|
|
if (bytes > 0)
|
|
{
|
|
if (!ack)
|
|
{
|
|
if ((bytes + inPos) <= inLen)
|
|
{
|
|
for (i=0; i<(bytes-1); i++)
|
|
{
|
|
ack = I2CPutByte(w, inBuf[inPos++]);
|
|
if (ack) status = PI_I2C_WRITE_FAILED;
|
|
}
|
|
ack = I2CPutByte(w, inBuf[inPos++]);
|
|
}
|
|
else status = PI_BAD_I2C_WLEN;
|
|
} else status = PI_I2C_WRITE_FAILED;
|
|
}
|
|
else status = PI_BAD_I2C_CMD;
|
|
break;
|
|
|
|
default:
|
|
status = PI_BAD_I2C_CMD;
|
|
}
|
|
|
|
if (setesc) esc = 1; else esc = 0;
|
|
|
|
setesc = 0;
|
|
}
|
|
|
|
wfRx_unlock(SDA);
|
|
|
|
if (status >= 0) status = outPos;
|
|
|
|
return status;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
void bscInit(int mode)
|
|
{
|
|
int sda, scl, mosi, miso, ce;
|
|
|
|
bscsReg[BSC_CR]=0; /* clear device */
|
|
bscsReg[BSC_RSR]=0; /* clear underrun and overrun errors */
|
|
bscsReg[BSC_SLV]=0; /* clear I2C slave address */
|
|
bscsReg[BSC_IMSC]=0xf; /* mask off all interrupts */
|
|
bscsReg[BSC_ICR]=0x0f; /* clear all interrupts */
|
|
|
|
if (pi_is_2711)
|
|
{
|
|
sda = BSC_SDA_2711;
|
|
scl = BSC_SCL_SCLK_2711;
|
|
mosi = BSC_MOSI_2711;
|
|
miso = BSC_MISO_2711;
|
|
ce = BSC_CE_N_2711;
|
|
}
|
|
else
|
|
{
|
|
sda = BSC_SDA;
|
|
scl = BSC_SCL_SCLK;
|
|
mosi = BSC_MOSI;
|
|
miso = BSC_MISO;
|
|
ce = BSC_CE_N;
|
|
}
|
|
|
|
|
|
if (mode > 1) /* SPI uses all GPIO */
|
|
{
|
|
gpioSetMode(scl, PI_ALT3);
|
|
gpioSetMode(mosi, PI_ALT3);
|
|
gpioSetMode(miso, PI_ALT3);
|
|
gpioSetMode(ce, PI_ALT3);
|
|
}
|
|
else
|
|
{
|
|
gpioSetMode(scl, PI_ALT3);
|
|
gpioSetMode(sda, PI_ALT3);
|
|
}
|
|
}
|
|
|
|
void bscTerm(int mode)
|
|
{
|
|
int sda, scl, mosi, miso, ce;
|
|
|
|
bscsReg[BSC_CR] = 0; /* clear device */
|
|
bscsReg[BSC_RSR]=0; /* clear underrun and overrun errors */
|
|
bscsReg[BSC_SLV]=0; /* clear I2C slave address */
|
|
|
|
if (pi_is_2711)
|
|
{
|
|
sda = BSC_SDA_2711;
|
|
scl = BSC_SCL_SCLK_2711;
|
|
mosi = BSC_MOSI_2711;
|
|
miso = BSC_MISO_2711;
|
|
ce = BSC_CE_N_2711;
|
|
}
|
|
else
|
|
{
|
|
sda = BSC_SDA;
|
|
scl = BSC_SCL_SCLK;
|
|
mosi = BSC_MOSI;
|
|
miso = BSC_MISO;
|
|
ce = BSC_CE_N;
|
|
}
|
|
|
|
|
|
if (mode > 1)
|
|
{
|
|
gpioSetMode(scl, PI_INPUT);
|
|
gpioSetMode(mosi, PI_INPUT);
|
|
gpioSetMode(miso, PI_INPUT);
|
|
gpioSetMode(ce, PI_INPUT);
|
|
}
|
|
else
|
|
{
|
|
gpioSetMode(sda, PI_INPUT);
|
|
gpioSetMode(scl, PI_INPUT);
|
|
|
|
}
|
|
}
|
|
|
|
int bscXfer(bsc_xfer_t *xfer)
|
|
{
|
|
static int bscMode = 0;
|
|
|
|
int copied=0;
|
|
int active, mode;
|
|
|
|
DBG(DBG_USER, "control=0x%X (sa=0x%X, cr=0x%X) tx=%d [%s]",
|
|
xfer->control,
|
|
((xfer->control)>>16) & 127,
|
|
(xfer->control) & 0x3fff,
|
|
xfer->txCnt,
|
|
myBuf2Str(xfer->txCnt, (char *)xfer->txBuf));
|
|
|
|
CHECK_INITED;
|
|
|
|
eventAlert[PI_EVENT_BSC].ignore = 1;
|
|
|
|
if (xfer->control)
|
|
{
|
|
/*
|
|
bscMode (0=None, 1=I2C, 2=SPI) tracks which GPIO have been
|
|
set to BSC mode
|
|
*/
|
|
if (xfer->control & 2) mode = 2; /* SPI */
|
|
else mode = 1; /* assume I2C */
|
|
|
|
if (mode > bscMode)
|
|
{
|
|
bscInit(mode);
|
|
bscMode = mode;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (bscMode) bscTerm(bscMode);
|
|
bscMode = 0;
|
|
return 0; /* leave ignore set */
|
|
}
|
|
|
|
xfer->rxCnt = 0;
|
|
|
|
bscsReg[BSC_SLV] = ((xfer->control)>>16) & 127;
|
|
bscsReg[BSC_CR] = (xfer->control) & 0x3fff;
|
|
bscsReg[BSC_RSR]=0; /* clear underrun and overrun errors */
|
|
|
|
active = 1;
|
|
|
|
while (active)
|
|
{
|
|
active = 0;
|
|
|
|
while ((copied < xfer->txCnt) &&
|
|
(!(bscsReg[BSC_FR] & BSC_FR_TXFF)))
|
|
{
|
|
bscsReg[BSC_DR] = xfer->txBuf[copied++];
|
|
active = 1;
|
|
}
|
|
|
|
while ((xfer->rxCnt < BSC_FIFO_SIZE) &&
|
|
(!(bscsReg[BSC_FR] & BSC_FR_RXFE)))
|
|
{
|
|
xfer->rxBuf[xfer->rxCnt++] = bscsReg[BSC_DR];
|
|
active = 1;
|
|
}
|
|
|
|
if (!active)
|
|
{
|
|
active = bscsReg[BSC_FR] & (BSC_FR_RXBUSY | BSC_FR_TXBUSY);
|
|
}
|
|
|
|
if (active) myGpioSleep(0, 20);
|
|
}
|
|
|
|
bscFR = bscsReg[BSC_FR] & 0xffff;
|
|
|
|
eventAlert[PI_EVENT_BSC].ignore = 0;
|
|
|
|
return (copied<<16) | bscFR;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static void set_CS(wfRx_t *w)
|
|
{
|
|
myGpioWrite(w->S.CS, PI_SPI_FLAGS_GET_CSPOL(w->S.spiFlags));
|
|
}
|
|
|
|
static void clear_CS(wfRx_t *w)
|
|
{
|
|
myGpioWrite(w->S.CS, !PI_SPI_FLAGS_GET_CSPOL(w->S.spiFlags));
|
|
}
|
|
|
|
static void set_SCLK(wfRx_t *w)
|
|
{
|
|
myGpioWrite(w->S.SCLK, !PI_SPI_FLAGS_GET_CPOL(w->S.spiFlags));
|
|
}
|
|
|
|
static void clear_SCLK(wfRx_t *w)
|
|
{
|
|
myGpioWrite(w->S.SCLK, PI_SPI_FLAGS_GET_CPOL(w->S.spiFlags));
|
|
}
|
|
|
|
static void SPI_delay(wfRx_t *w)
|
|
{
|
|
myGpioDelay(w->S.delay);
|
|
}
|
|
|
|
static void bbSPIStart(wfRx_t *w)
|
|
{
|
|
clear_SCLK(w);
|
|
|
|
SPI_delay(w);
|
|
|
|
set_CS(w);
|
|
|
|
SPI_delay(w);
|
|
}
|
|
|
|
static void bbSPIStop(wfRx_t *w)
|
|
{
|
|
SPI_delay(w);
|
|
|
|
clear_CS(w);
|
|
|
|
SPI_delay(w);
|
|
|
|
clear_SCLK(w);
|
|
}
|
|
|
|
static uint8_t bbSPIXferByte(wfRx_t *w, char txByte)
|
|
{
|
|
uint8_t bit, rxByte=0;
|
|
|
|
if (PI_SPI_FLAGS_GET_CPHA(w->S.spiFlags))
|
|
{
|
|
/*
|
|
CPHA = 1
|
|
write on set clock
|
|
read on clear clock
|
|
*/
|
|
|
|
for (bit=0; bit<8; bit++)
|
|
{
|
|
set_SCLK(w);
|
|
|
|
if (PI_SPI_FLAGS_GET_TX_LSB(w->S.spiFlags))
|
|
{
|
|
myGpioWrite(w->S.MOSI, txByte & 0x01);
|
|
txByte >>= 1;
|
|
}
|
|
else
|
|
{
|
|
myGpioWrite(w->S.MOSI, txByte & 0x80);
|
|
txByte <<= 1;
|
|
}
|
|
|
|
SPI_delay(w);
|
|
|
|
clear_SCLK(w);
|
|
|
|
if (PI_SPI_FLAGS_GET_RX_LSB(w->S.spiFlags))
|
|
{
|
|
rxByte = (rxByte >> 1) | myGpioRead(w->S.MISO) << 7;
|
|
}
|
|
else
|
|
{
|
|
rxByte = (rxByte << 1) | myGpioRead(w->S.MISO);
|
|
}
|
|
|
|
SPI_delay(w);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
/*
|
|
CPHA = 0
|
|
read on set clock
|
|
write on clear clock
|
|
*/
|
|
|
|
for (bit=0; bit<8; bit++)
|
|
{
|
|
if (PI_SPI_FLAGS_GET_TX_LSB(w->S.spiFlags))
|
|
{
|
|
myGpioWrite(w->S.MOSI, txByte & 0x01);
|
|
txByte >>= 1;
|
|
}
|
|
else
|
|
{
|
|
myGpioWrite(w->S.MOSI, txByte & 0x80);
|
|
txByte <<= 1;
|
|
}
|
|
|
|
SPI_delay(w);
|
|
|
|
set_SCLK(w);
|
|
|
|
if (PI_SPI_FLAGS_GET_RX_LSB(w->S.spiFlags))
|
|
{
|
|
rxByte = (rxByte >> 1) | myGpioRead(w->S.MISO) << 7;
|
|
}
|
|
else
|
|
{
|
|
rxByte = (rxByte << 1) | myGpioRead(w->S.MISO);
|
|
}
|
|
|
|
SPI_delay(w);
|
|
|
|
clear_SCLK(w);
|
|
}
|
|
}
|
|
|
|
return rxByte;
|
|
}
|
|
|
|
/*-------------------------------------------------------------------------*/
|
|
|
|
int bbSPIOpen(
|
|
unsigned CS, unsigned MISO, unsigned MOSI, unsigned SCLK,
|
|
unsigned baud, unsigned spiFlags)
|
|
{
|
|
int valid;
|
|
uint32_t bits;
|
|
|
|
DBG(DBG_USER, "CS=%d MISO=%d MOSI=%d SCLK=%d baud=%d flags=%d",
|
|
CS, MISO, MOSI, SCLK, baud, spiFlags);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (CS > PI_MAX_USER_GPIO)
|
|
SOFT_ERROR(PI_BAD_USER_GPIO, "bad CS (%d)", CS);
|
|
|
|
if (MISO > PI_MAX_USER_GPIO)
|
|
SOFT_ERROR(PI_BAD_USER_GPIO, "bad MISO (%d)", MISO);
|
|
|
|
if (MOSI > PI_MAX_USER_GPIO)
|
|
SOFT_ERROR(PI_BAD_USER_GPIO, "bad MOSI (%d)", MOSI);
|
|
|
|
if (SCLK > PI_MAX_USER_GPIO)
|
|
SOFT_ERROR(PI_BAD_USER_GPIO, "bad SCLK (%d)", SCLK);
|
|
|
|
if ((baud < PI_BB_SPI_MIN_BAUD) || (baud > PI_BB_SPI_MAX_BAUD))
|
|
SOFT_ERROR(PI_BAD_SPI_BAUD, "CS %d, bad baud (%d)", CS, baud);
|
|
|
|
if (wfRx[CS].mode != PI_WFRX_NONE)
|
|
SOFT_ERROR(PI_GPIO_IN_USE,
|
|
"CS %d is already being used, mode %d", CS, wfRx[CS].mode);
|
|
|
|
valid = 0;
|
|
|
|
/* check all GPIO unique */
|
|
|
|
bits = (1<<CS) | (1<<MISO) | (1<<MOSI) | (1<<SCLK);
|
|
|
|
if (__builtin_popcount(bits) == 4)
|
|
{
|
|
if ((wfRx[MISO].mode == PI_WFRX_NONE) &&
|
|
(wfRx[MOSI].mode == PI_WFRX_NONE) &&
|
|
(wfRx[SCLK].mode == PI_WFRX_NONE))
|
|
{
|
|
valid = 1; /* first time GPIO used for SPI */
|
|
}
|
|
else
|
|
{
|
|
if ((wfRx[MISO].mode == PI_WFRX_SPI_MISO) &&
|
|
(wfRx[MOSI].mode == PI_WFRX_SPI_MOSI) &&
|
|
(wfRx[SCLK].mode == PI_WFRX_SPI_SCLK))
|
|
{
|
|
valid = 2; /* new CS for existing SPI GPIO */
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!valid)
|
|
{
|
|
SOFT_ERROR(PI_GPIO_IN_USE,
|
|
"GPIO already being used (%d=%d %d=%d, %d=%d %d=%d)",
|
|
CS, wfRx[CS].mode,
|
|
MISO, wfRx[MISO].mode,
|
|
MOSI, wfRx[MOSI].mode,
|
|
SCLK, wfRx[SCLK].mode);
|
|
}
|
|
|
|
wfRx[CS].mode = PI_WFRX_SPI_CS;
|
|
wfRx[CS].baud = baud;
|
|
|
|
wfRx[CS].S.CS = CS;
|
|
wfRx[CS].S.SCLK = SCLK;
|
|
|
|
wfRx[CS].S.CSMode = gpioGetMode(CS);
|
|
wfRx[CS].S.delay = (500000 / baud) - 1;
|
|
wfRx[CS].S.spiFlags = spiFlags;
|
|
|
|
/* preset CS to off */
|
|
|
|
if (PI_SPI_FLAGS_GET_CSPOL(spiFlags))
|
|
gpioWrite(CS, 0); /* active high */
|
|
else
|
|
gpioWrite(CS, 1); /* active low */
|
|
|
|
/* The SCLK entry is used to store full information */
|
|
|
|
if (valid == 1) /* first time GPIO for SPI */
|
|
{
|
|
wfRx[SCLK].S.usage = 1;
|
|
|
|
wfRx[SCLK].S.SCLKMode = gpioGetMode(SCLK);
|
|
wfRx[SCLK].S.MISOMode = gpioGetMode(MISO);
|
|
wfRx[SCLK].S.MOSIMode = gpioGetMode(MOSI);
|
|
|
|
wfRx[SCLK].mode = PI_WFRX_SPI_SCLK;
|
|
wfRx[MISO].mode = PI_WFRX_SPI_MISO;
|
|
wfRx[MOSI].mode = PI_WFRX_SPI_MOSI;
|
|
|
|
wfRx[SCLK].S.SCLK = SCLK;
|
|
wfRx[SCLK].S.MISO = MISO;
|
|
wfRx[SCLK].S.MOSI = MOSI;
|
|
|
|
myGpioSetMode(MISO, PI_INPUT);
|
|
myGpioSetMode(SCLK, PI_OUTPUT);
|
|
gpioWrite(MOSI, 0); /* low output */
|
|
}
|
|
else
|
|
{
|
|
wfRx[SCLK].S.usage++;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*-------------------------------------------------------------------------*/
|
|
|
|
int bbSPIClose(unsigned CS)
|
|
{
|
|
int SCLK;
|
|
|
|
DBG(DBG_USER, "CS=%d", CS);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (CS > PI_MAX_USER_GPIO)
|
|
SOFT_ERROR(PI_BAD_USER_GPIO, "bad gpio (%d)", CS);
|
|
|
|
switch(wfRx[CS].mode)
|
|
{
|
|
case PI_WFRX_SPI_CS:
|
|
|
|
myGpioSetMode(wfRx[CS].S.CS, wfRx[CS].S.CSMode);
|
|
wfRx[CS].mode = PI_WFRX_NONE;
|
|
|
|
SCLK = wfRx[CS].S.SCLK;
|
|
|
|
if (--wfRx[SCLK].S.usage <= 0)
|
|
{
|
|
myGpioSetMode(wfRx[SCLK].S.MISO, wfRx[SCLK].S.MISOMode);
|
|
myGpioSetMode(wfRx[SCLK].S.MOSI, wfRx[SCLK].S.MOSIMode);
|
|
myGpioSetMode(wfRx[SCLK].S.SCLK, wfRx[SCLK].S.SCLKMode);
|
|
|
|
wfRx[wfRx[SCLK].S.MISO].mode = PI_WFRX_NONE;
|
|
wfRx[wfRx[SCLK].S.MOSI].mode = PI_WFRX_NONE;
|
|
wfRx[wfRx[SCLK].S.SCLK].mode = PI_WFRX_NONE;
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
SOFT_ERROR(PI_NOT_SPI_GPIO, "no SPI on gpio (%d)", CS);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*-------------------------------------------------------------------------*/
|
|
|
|
int bbSPIXfer(
|
|
unsigned CS,
|
|
char *inBuf,
|
|
char *outBuf,
|
|
unsigned count)
|
|
{
|
|
int SCLK;
|
|
int pos;
|
|
wfRx_t *w;
|
|
|
|
DBG(DBG_USER, "CS=%d inBuf=%s outBuf=%08"PRIXPTR" count=%d",
|
|
CS, myBuf2Str(count, (char *)inBuf), (uintptr_t)outBuf, count);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (CS > PI_MAX_USER_GPIO)
|
|
SOFT_ERROR(PI_BAD_USER_GPIO, "bad gpio (%d)", CS);
|
|
|
|
if (wfRx[CS].mode != PI_WFRX_SPI_CS)
|
|
SOFT_ERROR(PI_NOT_SPI_GPIO, "no SPI on gpio (%d)", CS);
|
|
|
|
if (!inBuf || !count)
|
|
SOFT_ERROR(PI_BAD_POINTER, "input buffer can't be NULL");
|
|
|
|
if (!outBuf && count)
|
|
SOFT_ERROR(PI_BAD_POINTER, "output buffer can't be NULL");
|
|
|
|
SCLK = wfRx[CS].S.SCLK;
|
|
|
|
wfRx[SCLK].S.CS = CS;
|
|
wfRx[SCLK].baud = wfRx[CS].baud;
|
|
wfRx[SCLK].S.delay = wfRx[CS].S.delay;
|
|
wfRx[SCLK].S.spiFlags = wfRx[CS].S.spiFlags;
|
|
|
|
w = &wfRx[SCLK];
|
|
|
|
wfRx_lock(SCLK);
|
|
|
|
bbSPIStart(w);
|
|
|
|
for (pos=0; pos < count; pos++)
|
|
{
|
|
outBuf[pos] = bbSPIXferByte(w, inBuf[pos]);
|
|
}
|
|
|
|
bbSPIStop(w);
|
|
|
|
wfRx_unlock(SCLK);
|
|
|
|
return count;
|
|
}
|
|
|
|
/*-------------------------------------------------------------------------*/
|
|
|
|
int gpioSerialReadOpen(unsigned gpio, unsigned baud, unsigned data_bits)
|
|
{
|
|
int bitTime, timeout;
|
|
|
|
DBG(DBG_USER, "gpio=%d baud=%d data_bits=%d", gpio, baud, data_bits);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (gpio > PI_MAX_USER_GPIO)
|
|
SOFT_ERROR(PI_BAD_USER_GPIO, "bad gpio (%d)", gpio);
|
|
|
|
if ((baud < PI_BB_SER_MIN_BAUD) || (baud > PI_BB_SER_MAX_BAUD))
|
|
SOFT_ERROR(PI_BAD_WAVE_BAUD,
|
|
"gpio %d, bad baud rate (%d)", gpio, baud);
|
|
|
|
if ((data_bits < PI_MIN_WAVE_DATABITS) ||
|
|
(data_bits > PI_MAX_WAVE_DATABITS))
|
|
SOFT_ERROR(PI_BAD_DATABITS,
|
|
"gpio %d, bad data bits (%d)", gpio, data_bits);
|
|
|
|
if (wfRx[gpio].mode != PI_WFRX_NONE)
|
|
SOFT_ERROR(PI_GPIO_IN_USE, "gpio %d is already being used", gpio);
|
|
|
|
bitTime = (1000 * MILLION) / baud; /* nanos */
|
|
|
|
timeout = ((data_bits+2) * bitTime)/MILLION; /* millis */
|
|
|
|
if (timeout < 1) timeout = 1;
|
|
|
|
wfRx[gpio].gpio = gpio;
|
|
wfRx[gpio].mode = PI_WFRX_SERIAL;
|
|
wfRx[gpio].baud = baud;
|
|
|
|
wfRx[gpio].s.buf = malloc(SRX_BUF_SIZE);
|
|
wfRx[gpio].s.bufSize = SRX_BUF_SIZE;
|
|
wfRx[gpio].s.timeout = timeout;
|
|
wfRx[gpio].s.fullBit = bitTime; /* nanos */
|
|
wfRx[gpio].s.halfBit = (bitTime/2)+500; /* nanos (500 for rounding) */
|
|
wfRx[gpio].s.readPos = 0;
|
|
wfRx[gpio].s.writePos = 0;
|
|
wfRx[gpio].s.bit = -1;
|
|
wfRx[gpio].s.dataBits = data_bits;
|
|
wfRx[gpio].s.invert = PI_BB_SER_NORMAL;
|
|
|
|
if (data_bits < 9) wfRx[gpio].s.bytes = 1;
|
|
else if (data_bits < 17) wfRx[gpio].s.bytes = 2;
|
|
else wfRx[gpio].s.bytes = 4;
|
|
|
|
gpioSetAlertFunc(gpio, waveRxBit);
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*-------------------------------------------------------------------------*/
|
|
|
|
int gpioSerialReadInvert(unsigned gpio, unsigned invert)
|
|
{
|
|
DBG(DBG_USER, "gpio=%d invert=%d", gpio, invert);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (gpio > PI_MAX_USER_GPIO)
|
|
SOFT_ERROR(PI_BAD_USER_GPIO, "bad gpio (%d)", gpio);
|
|
|
|
if (wfRx[gpio].mode != PI_WFRX_SERIAL)
|
|
SOFT_ERROR(PI_NOT_SERIAL_GPIO, "no serial read on gpio (%d)", gpio);
|
|
|
|
if ((invert < PI_BB_SER_NORMAL) ||
|
|
(invert > PI_BB_SER_INVERT))
|
|
SOFT_ERROR(PI_BAD_SER_INVERT,
|
|
"bad invert level for gpio %d (%d)", gpio, invert);
|
|
|
|
wfRx[gpio].s.invert = invert;
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*-------------------------------------------------------------------------*/
|
|
|
|
int gpioSerialRead(unsigned gpio, void *buf, size_t bufSize)
|
|
{
|
|
unsigned bytes=0, wpos;
|
|
volatile wfRx_t *w;
|
|
|
|
DBG(DBG_USER, "gpio=%d buf=%08"PRIXPTR" bufSize=%zd", gpio, (uintptr_t)buf, bufSize);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (gpio > PI_MAX_USER_GPIO)
|
|
SOFT_ERROR(PI_BAD_USER_GPIO, "bad gpio (%d)", gpio);
|
|
|
|
if (bufSize == 0)
|
|
SOFT_ERROR(PI_BAD_SERIAL_COUNT, "buffer size can't be zero");
|
|
|
|
if (wfRx[gpio].mode != PI_WFRX_SERIAL)
|
|
SOFT_ERROR(PI_NOT_SERIAL_GPIO, "no serial read on gpio (%d)", gpio);
|
|
|
|
w = &wfRx[gpio];
|
|
|
|
if (w->s.readPos != w->s.writePos)
|
|
{
|
|
wpos = w->s.writePos;
|
|
|
|
if (wpos > w->s.readPos) bytes = wpos - w->s.readPos;
|
|
else bytes = w->s.bufSize - w->s.readPos;
|
|
|
|
if (bytes > bufSize) bytes = bufSize;
|
|
|
|
/* copy in multiples of the data size in bytes */
|
|
|
|
bytes = (bytes / w->s.bytes) * w->s.bytes;
|
|
|
|
if (buf) memcpy(buf, w->s.buf+w->s.readPos, bytes);
|
|
|
|
w->s.readPos += bytes;
|
|
|
|
if (w->s.readPos >= w->s.bufSize) w->s.readPos = 0;
|
|
}
|
|
return bytes;
|
|
}
|
|
|
|
|
|
/*-------------------------------------------------------------------------*/
|
|
|
|
int gpioSerialReadClose(unsigned gpio)
|
|
{
|
|
DBG(DBG_USER, "gpio=%d", gpio);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (gpio > PI_MAX_USER_GPIO)
|
|
SOFT_ERROR(PI_BAD_USER_GPIO, "bad gpio (%d)", gpio);
|
|
|
|
switch(wfRx[gpio].mode)
|
|
{
|
|
case PI_WFRX_NONE:
|
|
|
|
SOFT_ERROR(PI_NOT_SERIAL_GPIO, "no serial read on gpio (%d)", gpio);
|
|
|
|
break;
|
|
|
|
case PI_WFRX_SERIAL:
|
|
|
|
free(wfRx[gpio].s.buf);
|
|
|
|
gpioSetWatchdog(gpio, 0); /* switch off timeouts */
|
|
|
|
gpioSetAlertFunc(gpio, NULL); /* cancel alert */
|
|
|
|
wfRx[gpio].mode = PI_WFRX_NONE;
|
|
|
|
break;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static int intEventSetFunc(
|
|
unsigned event,
|
|
void * f,
|
|
int user,
|
|
void * userdata)
|
|
{
|
|
DBG(DBG_INTERNAL, "event=%d function=%08"PRIXPTR", user=%d, userdata=%08"PRIXPTR,
|
|
event, (uintptr_t)f, user, (uintptr_t)userdata);
|
|
|
|
eventAlert[event].ex = user;
|
|
eventAlert[event].userdata = userdata;
|
|
|
|
eventAlert[event].func = f;
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int eventSetFunc(unsigned event, eventFunc_t f)
|
|
{
|
|
DBG(DBG_USER, "event=%d function=%08"PRIXPTR, event, (uintptr_t)f);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (event > PI_MAX_EVENT)
|
|
SOFT_ERROR(PI_BAD_EVENT_ID, "bad event (%d)", event);
|
|
|
|
intEventSetFunc(event, f, 0, NULL);
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int eventSetFuncEx(unsigned event, eventFuncEx_t f, void *userdata)
|
|
{
|
|
DBG(DBG_USER, "event=%d function=%08"PRIxPTR" userdata=%08"PRIxPTR,
|
|
event, (uintptr_t)f, (uintptr_t)userdata);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (event > PI_MAX_EVENT)
|
|
SOFT_ERROR(PI_BAD_EVENT_ID, "bad event (%d)", event);
|
|
|
|
intEventSetFunc(event, f, 1, userdata);
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int eventMonitor(unsigned handle, uint32_t bits)
|
|
{
|
|
DBG(DBG_USER, "handle=%d bits=%08X", handle, bits);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (handle >= PI_NOTIFY_SLOTS)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
if (gpioNotify[handle].state <= PI_NOTIFY_CLOSING)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
gpioNotify[handle].eventBits = bits;
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int eventTrigger(unsigned event)
|
|
{
|
|
DBG(DBG_USER, "event=%d", event);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (event > PI_MAX_EVENT)
|
|
SOFT_ERROR(PI_BAD_EVENT_ID, "bad event (%d)", event);
|
|
|
|
eventAlert[event].fired = 1;
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static int intGpioSetAlertFunc(
|
|
unsigned gpio,
|
|
void * f,
|
|
int user,
|
|
void * userdata)
|
|
{
|
|
DBG(DBG_INTERNAL, "gpio=%d function=%08"PRIXPTR", user=%d, userdata=%08"PRIXPTR,
|
|
gpio, (uintptr_t)f, user, (uintptr_t)userdata);
|
|
|
|
gpioAlert[gpio].ex = user;
|
|
gpioAlert[gpio].userdata = userdata;
|
|
|
|
gpioAlert[gpio].func = f;
|
|
|
|
if (f)
|
|
{
|
|
alertBits |= BIT;
|
|
}
|
|
else
|
|
{
|
|
alertBits &= ~BIT;
|
|
}
|
|
|
|
monitorBits = alertBits | notifyBits | scriptBits | gpioGetSamples.bits;
|
|
|
|
return 0;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioSetAlertFunc(unsigned gpio, gpioAlertFunc_t f)
|
|
{
|
|
DBG(DBG_USER, "gpio=%d function=%08"PRIXPTR, gpio, (uintptr_t)f);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (gpio > PI_MAX_USER_GPIO)
|
|
SOFT_ERROR(PI_BAD_USER_GPIO, "bad gpio (%d)", gpio);
|
|
|
|
intGpioSetAlertFunc(gpio, f, 0, NULL);
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioSetAlertFuncEx(unsigned gpio, gpioAlertFuncEx_t f, void *userdata)
|
|
{
|
|
DBG(DBG_USER, "gpio=%d function=%08"PRIXPTR" userdata=%08"PRIXPTR,
|
|
gpio, (uintptr_t)f, (uintptr_t)userdata);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (gpio > PI_MAX_USER_GPIO)
|
|
SOFT_ERROR(PI_BAD_USER_GPIO, "bad gpio (%d)", gpio);
|
|
|
|
intGpioSetAlertFunc(gpio, f, 1, userdata);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void *pthISRThread(void *x)
|
|
{
|
|
gpioISR_t *isr = x;
|
|
int fd;
|
|
int retval;
|
|
uint32_t tick;
|
|
int level;
|
|
uint32_t levels;
|
|
struct pollfd pfd;
|
|
char buf[64];
|
|
|
|
DBG(DBG_USER, "gpio=%d edge=%d timeout=%d f=%"PRIxPTR" u=%d data=%"PRIxPTR,
|
|
isr->gpio, isr->edge, isr->timeout, (uintptr_t)isr->func,
|
|
isr->ex, (uintptr_t)isr->userdata);
|
|
|
|
sprintf(buf, "/sys/class/gpio/gpio%d/value", isr->gpio);
|
|
|
|
isr->fd = -1; /* no fd assigned */
|
|
|
|
if ((fd = open(buf, O_RDONLY)) < 0)
|
|
{
|
|
DBG(DBG_ALWAYS, "gpio %d not exported", isr->gpio);
|
|
return NULL;
|
|
}
|
|
|
|
isr->fd = fd; /* store fd so it can be closed */
|
|
|
|
pfd.fd = fd;
|
|
|
|
pfd.events = POLLPRI;
|
|
|
|
lseek(fd, 0, SEEK_SET); /* consume any prior interrupt */
|
|
if (read(fd, buf, sizeof buf) == -1) { /* ignore errors */ }
|
|
|
|
while (1)
|
|
{
|
|
retval = poll(&pfd, 1, isr->timeout); /* wait for interrupt */
|
|
|
|
tick = systReg[SYST_CLO];
|
|
|
|
levels = *(gpioReg + GPLEV0);
|
|
|
|
if (retval >= 0)
|
|
{
|
|
lseek(fd, 0, SEEK_SET); /* consume interrupt */
|
|
if (read(fd, buf, sizeof buf) == -1) { /* ignore errors */ }
|
|
|
|
if (retval)
|
|
{
|
|
if (levels & (1<<isr->gpio)) level = PI_ON; else level = PI_OFF;
|
|
}
|
|
else level = PI_TIMEOUT;
|
|
|
|
if (isr->ex) (isr->func)(isr->gpio, level, tick, isr->userdata);
|
|
else (isr->func)(isr->gpio, level, tick);
|
|
}
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static int intGpioSetISRFunc(
|
|
unsigned gpio,
|
|
unsigned edge,
|
|
int timeout,
|
|
void *f,
|
|
int user,
|
|
void *userdata)
|
|
{
|
|
char buf[64];
|
|
|
|
char *edge_str[]={"rising\n", "falling\n", "both\n"};
|
|
int fd;
|
|
int err;
|
|
|
|
DBG(DBG_INTERNAL,
|
|
"gpio=%d edge=%d timeout=%d function=%08"PRIXPTR" user=%d userdata=%08"PRIXPTR,
|
|
gpio, edge, timeout, (uintptr_t)f, user, (uintptr_t)userdata);
|
|
|
|
if (f)
|
|
{
|
|
if (!gpioISR[gpio].inited) /* export gpio if unexported */
|
|
{
|
|
fd = open("/sys/class/gpio/export", O_WRONLY);
|
|
if (fd < 0) return PI_BAD_ISR_INIT;
|
|
|
|
/* ignore write fail if already exported */
|
|
sprintf(buf, "%d\n", gpio);
|
|
err = write(fd, buf, strlen(buf));
|
|
close(fd);
|
|
|
|
sprintf(buf, "/sys/class/gpio/gpio%d/direction", gpio);
|
|
fd = open(buf, O_WRONLY);
|
|
if (fd < 0) return PI_BAD_ISR_INIT;
|
|
|
|
err = write(fd, "in\n", 3);
|
|
close(fd);
|
|
if (err != 3) return PI_BAD_ISR_INIT;
|
|
|
|
gpioISR[gpio].gpio = gpio;
|
|
gpioISR[gpio].edge = -1;
|
|
gpioISR[gpio].timeout = -1;
|
|
|
|
gpioISR[gpio].inited = 1;
|
|
}
|
|
|
|
if (gpioISR[gpio].edge != edge)
|
|
{
|
|
sprintf(buf, "/sys/class/gpio/gpio%d/edge", gpio);
|
|
fd = open(buf, O_WRONLY);
|
|
if (fd < 0) return PI_BAD_ISR_INIT;
|
|
|
|
err = write(fd, edge_str[edge], strlen(edge_str[edge]));
|
|
close(fd);
|
|
if (err != strlen(edge_str[edge])) return PI_BAD_ISR_INIT;
|
|
|
|
gpioISR[gpio].edge = edge;
|
|
|
|
if (gpioISR[gpio].pth != NULL)
|
|
pthread_kill(*gpioISR[gpio].pth, SIGCHLD);
|
|
}
|
|
|
|
if (timeout <= 0) timeout = -1;
|
|
if (gpioISR[gpio].timeout != timeout)
|
|
{
|
|
gpioISR[gpio].timeout = timeout;
|
|
|
|
if (gpioISR[gpio].pth != NULL)
|
|
pthread_kill(*gpioISR[gpio].pth, SIGCHLD);
|
|
}
|
|
|
|
gpioISR[gpio].func = f;
|
|
gpioISR[gpio].ex = user;
|
|
gpioISR[gpio].userdata = userdata;
|
|
|
|
if (gpioISR[gpio].pth == NULL)
|
|
gpioISR[gpio].pth = gpioStartThread(pthISRThread, &gpioISR[gpio]);
|
|
}
|
|
else /* null function, delete ISR, unexport gpio */
|
|
{
|
|
if (gpioISR[gpio].pth) /* delete any existing ISR */
|
|
{
|
|
gpioStopThread(gpioISR[gpio].pth);
|
|
|
|
if (gpioISR[gpio].fd >= 0)
|
|
{
|
|
close(gpioISR[gpio].fd);
|
|
gpioISR[gpio].fd = -1;
|
|
}
|
|
|
|
gpioISR[gpio].func = NULL;
|
|
gpioISR[gpio].pth = NULL;
|
|
}
|
|
|
|
if (gpioISR[gpio].inited) /* unexport the gpio */
|
|
{
|
|
fd = open("/sys/class/gpio/unexport", O_WRONLY);
|
|
if (fd < 0) return PI_BAD_ISR_INIT;
|
|
sprintf(buf, "%d\n", gpio);
|
|
err = write(fd, buf, strlen(buf));
|
|
close(fd);
|
|
if (err != strlen(buf)) return PI_BAD_ISR_INIT;
|
|
gpioISR[gpio].inited = 0;
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioSetISRFunc(
|
|
unsigned gpio,
|
|
unsigned edge,
|
|
int timeout,
|
|
gpioISRFunc_t f)
|
|
{
|
|
DBG(DBG_USER, "gpio=%d edge=%d timeout=%d function=%08"PRIXPTR,
|
|
gpio, edge, timeout, (uintptr_t)f);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (gpio > PI_MAX_GPIO)
|
|
SOFT_ERROR(PI_BAD_GPIO, "bad gpio (%d)", gpio);
|
|
|
|
if (edge > EITHER_EDGE)
|
|
SOFT_ERROR(PI_BAD_EDGE, "bad ISR edge (%d)", edge);
|
|
|
|
return intGpioSetISRFunc(gpio, edge, timeout, f, 0, NULL);
|
|
}
|
|
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioSetISRFuncEx(
|
|
unsigned gpio,
|
|
unsigned edge,
|
|
int timeout,
|
|
gpioAlertFuncEx_t f,
|
|
void *userdata)
|
|
{
|
|
DBG(DBG_USER, "gpio=%d edge=%d timeout=%d function=%08"PRIXPTR" userdata=%08"PRIXPTR,
|
|
gpio, edge, timeout, (uintptr_t)f, (uintptr_t)userdata);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (gpio > PI_MAX_GPIO)
|
|
SOFT_ERROR(PI_BAD_GPIO, "bad gpio (%d)", gpio);
|
|
|
|
if (edge > EITHER_EDGE)
|
|
SOFT_ERROR(PI_BAD_EDGE, "bad ISR edge (%d)", edge);
|
|
|
|
return intGpioSetISRFunc(gpio, edge, timeout, f, 1, userdata);
|
|
}
|
|
|
|
static void closeOrphanedNotifications(int slot, int fd)
|
|
{
|
|
int i;
|
|
|
|
/* Check for and close any orphaned notifications. */
|
|
|
|
for (i=0; i<PI_NOTIFY_SLOTS; i++)
|
|
{
|
|
if ((i != slot) &&
|
|
(gpioNotify[i].state >= PI_NOTIFY_OPENED) &&
|
|
(gpioNotify[i].fd == fd))
|
|
{
|
|
DBG(DBG_USER, "closed orphaned fd=%d (handle=%d)", fd, i);
|
|
gpioNotify[i].state = PI_NOTIFY_CLOSED;
|
|
intNotifyBits();
|
|
}
|
|
}
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static void notifyMutex(int lock)
|
|
{
|
|
static pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER;
|
|
if (lock) pthread_mutex_lock(&mutex);
|
|
else pthread_mutex_unlock(&mutex);
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioNotifyOpenWithSize(int bufSize)
|
|
{
|
|
int i, slot, fd;
|
|
char name[32];
|
|
|
|
DBG(DBG_USER, "bufSize=%d", bufSize);
|
|
|
|
CHECK_INITED;
|
|
|
|
slot = -1;
|
|
|
|
notifyMutex(1);
|
|
|
|
for (i=0; i<PI_NOTIFY_SLOTS; i++)
|
|
{
|
|
if (gpioNotify[i].state == PI_NOTIFY_CLOSED)
|
|
{
|
|
slot = i;
|
|
gpioNotify[slot].state = PI_NOTIFY_RESERVED;
|
|
break;
|
|
}
|
|
}
|
|
|
|
notifyMutex(0);
|
|
|
|
if (slot < 0) SOFT_ERROR(PI_NO_HANDLE, "no handle");
|
|
|
|
sprintf(name, "/dev/pigpio%d", slot);
|
|
|
|
myCreatePipe(name, 0664);
|
|
|
|
fd = open(name, O_RDWR|O_NONBLOCK);
|
|
|
|
if (fd < 0)
|
|
{
|
|
gpioNotify[slot].state = PI_NOTIFY_CLOSED;
|
|
SOFT_ERROR(PI_BAD_PATHNAME, "open %s failed (%m)", name);
|
|
}
|
|
|
|
if (bufSize != 0)
|
|
{
|
|
i = fcntl(fd, F_SETPIPE_SZ, bufSize);
|
|
if (i != bufSize)
|
|
{
|
|
gpioNotify[slot].state = PI_NOTIFY_CLOSED;
|
|
SOFT_ERROR(PI_BAD_PATHNAME,
|
|
"fcntl %s size %d failed (%m)", name, bufSize);
|
|
}
|
|
}
|
|
|
|
gpioNotify[slot].seqno = 0;
|
|
gpioNotify[slot].bits = 0;
|
|
gpioNotify[slot].fd = fd;
|
|
gpioNotify[slot].pipe = 1;
|
|
gpioNotify[slot].max_emits = MAX_EMITS;
|
|
gpioNotify[slot].lastReportTick = gpioTick();
|
|
gpioNotify[slot].state = PI_NOTIFY_OPENED;
|
|
|
|
closeOrphanedNotifications(slot, fd);
|
|
|
|
return slot;
|
|
}
|
|
|
|
int gpioNotifyOpen(void)
|
|
{
|
|
return gpioNotifyOpenWithSize(0);
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static int gpioNotifyOpenInBand(int fd)
|
|
{
|
|
int i, slot;
|
|
|
|
DBG(DBG_USER, "fd=%d", fd);
|
|
|
|
CHECK_INITED;
|
|
|
|
slot = -1;
|
|
|
|
notifyMutex(1);
|
|
|
|
for (i=0; i<PI_NOTIFY_SLOTS; i++)
|
|
{
|
|
if (gpioNotify[i].state == PI_NOTIFY_CLOSED)
|
|
{
|
|
slot = i;
|
|
gpioNotify[slot].state = PI_NOTIFY_RESERVED;
|
|
break;
|
|
}
|
|
}
|
|
|
|
notifyMutex(0);
|
|
|
|
if (slot < 0) SOFT_ERROR(PI_NO_HANDLE, "no handle");
|
|
|
|
gpioNotify[slot].seqno = 0;
|
|
gpioNotify[slot].bits = 0;
|
|
gpioNotify[slot].fd = fd;
|
|
gpioNotify[slot].pipe = 0;
|
|
gpioNotify[slot].max_emits = MAX_EMITS;
|
|
gpioNotify[slot].lastReportTick = gpioTick();
|
|
gpioNotify[slot].state = PI_NOTIFY_OPENED;
|
|
|
|
closeOrphanedNotifications(slot, fd);
|
|
|
|
return slot;
|
|
}
|
|
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static void intScriptBits(void)
|
|
{
|
|
int i;
|
|
uint32_t bits;
|
|
|
|
bits = 0;
|
|
|
|
for (i=0; i<PI_MAX_SCRIPTS; i++)
|
|
{
|
|
if (gpioScript[i].state == PI_SCRIPT_IN_USE)
|
|
{
|
|
bits |= gpioScript[i].waitBits;
|
|
}
|
|
}
|
|
|
|
scriptBits = bits;
|
|
|
|
monitorBits = alertBits | notifyBits | scriptBits | gpioGetSamples.bits;
|
|
}
|
|
|
|
|
|
static void intScriptEventBits(void)
|
|
{
|
|
int i;
|
|
uint32_t bits;
|
|
|
|
bits = 0;
|
|
|
|
for (i=0; i<PI_MAX_SCRIPTS; i++)
|
|
{
|
|
if (gpioScript[i].state == PI_SCRIPT_IN_USE)
|
|
{
|
|
bits |= gpioScript[i].eventBits;
|
|
}
|
|
}
|
|
|
|
scriptEventBits = bits;
|
|
}
|
|
|
|
|
|
static void intNotifyBits(void)
|
|
{
|
|
int i;
|
|
uint32_t bits;
|
|
|
|
bits = 0;
|
|
|
|
for (i=0; i<PI_NOTIFY_SLOTS; i++)
|
|
{
|
|
if (gpioNotify[i].state == PI_NOTIFY_RUNNING)
|
|
{
|
|
bits |= gpioNotify[i].bits;
|
|
}
|
|
}
|
|
|
|
notifyBits = bits;
|
|
|
|
monitorBits = alertBits | notifyBits | scriptBits | gpioGetSamples.bits;
|
|
}
|
|
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioNotifyBegin(unsigned handle, uint32_t bits)
|
|
{
|
|
DBG(DBG_USER, "handle=%d bits=%08X", handle, bits);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (handle >= PI_NOTIFY_SLOTS)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
if (gpioNotify[handle].state <= PI_NOTIFY_CLOSING)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
gpioNotify[handle].bits = bits;
|
|
|
|
gpioNotify[handle].state = PI_NOTIFY_RUNNING;
|
|
|
|
intNotifyBits();
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioNotifyPause (unsigned handle)
|
|
{
|
|
DBG(DBG_USER, "handle=%d", handle);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (handle >= PI_NOTIFY_SLOTS)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
if (gpioNotify[handle].state <= PI_NOTIFY_CLOSING)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
gpioNotify[handle].bits = 0;
|
|
|
|
gpioNotify[handle].state = PI_NOTIFY_PAUSED;
|
|
|
|
intNotifyBits();
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioNotifyClose(unsigned handle)
|
|
{
|
|
char fifo[32];
|
|
|
|
DBG(DBG_USER, "handle=%d", handle);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (handle >= PI_NOTIFY_SLOTS)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
if (gpioNotify[handle].state <= PI_NOTIFY_CLOSING)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
gpioNotify[handle].bits = 0;
|
|
|
|
gpioNotify[handle].state = PI_NOTIFY_CLOSING;
|
|
|
|
intNotifyBits();
|
|
|
|
if (gpioCfg.ifFlags & PI_DISABLE_ALERT)
|
|
{
|
|
if (gpioNotify[handle].pipe)
|
|
{
|
|
DBG(DBG_INTERNAL, "close notify pipe %d", gpioNotify[handle].fd);
|
|
close(gpioNotify[handle].fd);
|
|
|
|
sprintf(fifo, "/dev/pigpio%d", handle);
|
|
|
|
unlink(fifo);
|
|
}
|
|
|
|
gpioNotify[handle].state = PI_NOTIFY_CLOSED;
|
|
}
|
|
else
|
|
{
|
|
/* actual close done in alert thread */
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioTrigger(unsigned gpio, unsigned pulseLen, unsigned level)
|
|
{
|
|
DBG(DBG_USER, "gpio=%d pulseLen=%d level=%d", gpio, pulseLen, level);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (gpio > PI_MAX_USER_GPIO)
|
|
SOFT_ERROR(PI_BAD_USER_GPIO, "bad gpio (%d)", gpio);
|
|
|
|
if (level > PI_ON)
|
|
SOFT_ERROR(PI_BAD_LEVEL, "gpio %d, bad level (%d)", gpio, level);
|
|
|
|
if ((pulseLen > PI_MAX_BUSY_DELAY) || (!pulseLen))
|
|
SOFT_ERROR(PI_BAD_PULSELEN,
|
|
"gpio %d, bad pulseLen (%d)", gpio, pulseLen);
|
|
|
|
if (level == PI_OFF) *(gpioReg + GPCLR0 + BANK) = BIT;
|
|
else *(gpioReg + GPSET0 + BANK) = BIT;
|
|
|
|
myGpioDelay(pulseLen);
|
|
|
|
if (level != PI_OFF) *(gpioReg + GPCLR0 + BANK) = BIT;
|
|
else *(gpioReg + GPSET0 + BANK) = BIT;
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioSetWatchdog(unsigned gpio, unsigned timeout)
|
|
{
|
|
DBG(DBG_USER, "gpio=%d timeout=%d", gpio, timeout);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (gpio > PI_MAX_USER_GPIO)
|
|
SOFT_ERROR(PI_BAD_USER_GPIO, "bad gpio (%d)", gpio);
|
|
|
|
if (timeout > PI_MAX_WDOG_TIMEOUT)
|
|
SOFT_ERROR(PI_BAD_WDOG_TIMEOUT,
|
|
"gpio %d, bad timeout (%d)", gpio, timeout);
|
|
|
|
gpioAlert[gpio].wdTick = systReg[SYST_CLO];
|
|
gpioAlert[gpio].wdSteadyUs = timeout*1000;
|
|
|
|
if (timeout) wdogBits |= (1<<gpio);
|
|
else wdogBits &= (~(1<<gpio));
|
|
|
|
return 0;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioNoiseFilter(unsigned gpio, unsigned steady, unsigned active)
|
|
{
|
|
DBG(DBG_USER, "gpio=%d steady=%d active=%d", gpio, steady, active);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (gpio > PI_MAX_USER_GPIO)
|
|
SOFT_ERROR(PI_BAD_USER_GPIO, "bad gpio (%d)", gpio);
|
|
|
|
if (steady > PI_MAX_STEADY)
|
|
SOFT_ERROR(PI_BAD_FILTER, "bad steady (%d)", steady);
|
|
|
|
if (active > PI_MAX_ACTIVE)
|
|
SOFT_ERROR(PI_BAD_FILTER, "bad active (%d)", active);
|
|
|
|
gpioAlert[gpio].nfTick1 = systReg[SYST_CLO];
|
|
gpioAlert[gpio].nfTick2 = gpioAlert[gpio].nfTick1;
|
|
gpioAlert[gpio].nfSteadyUs = steady;
|
|
gpioAlert[gpio].nfActiveUs = active;
|
|
gpioAlert[gpio].nfActive = 0;
|
|
|
|
if (steady) nFilterBits |= (1<<gpio);
|
|
else nFilterBits &= (~(1<<gpio));
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioGlitchFilter(unsigned gpio, unsigned steady)
|
|
{
|
|
DBG(DBG_USER, "gpio=%d steady=%d", gpio, steady);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (gpio > PI_MAX_USER_GPIO)
|
|
SOFT_ERROR(PI_BAD_USER_GPIO, "bad gpio (%d)", gpio);
|
|
|
|
if (steady > PI_MAX_STEADY)
|
|
SOFT_ERROR(PI_BAD_FILTER, "bad steady (%d)", steady);
|
|
|
|
if (steady)
|
|
{
|
|
/* Initialise values next time we process alerts */
|
|
gpioAlert[gpio].gfInitialised = 0;
|
|
}
|
|
|
|
gpioAlert[gpio].gfSteadyUs = steady;
|
|
|
|
if (steady) gFilterBits |= (1<<gpio);
|
|
else gFilterBits &= (~(1<<gpio));
|
|
|
|
return 0;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioSetGetSamplesFunc(gpioGetSamplesFunc_t f, uint32_t bits)
|
|
{
|
|
DBG(DBG_USER, "function=%08"PRIXPTR" bits=%08X", (uintptr_t)f, bits);
|
|
|
|
CHECK_INITED;
|
|
|
|
gpioGetSamples.ex = 0;
|
|
gpioGetSamples.userdata = NULL;
|
|
gpioGetSamples.func = f;
|
|
|
|
if (f) gpioGetSamples.bits = bits;
|
|
else gpioGetSamples.bits = 0;
|
|
|
|
monitorBits = alertBits | notifyBits | scriptBits | gpioGetSamples.bits;
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioSetGetSamplesFuncEx(gpioGetSamplesFuncEx_t f,
|
|
uint32_t bits,
|
|
void * userdata)
|
|
{
|
|
DBG(DBG_USER, "function=%08"PRIXPTR" bits=%08X", (uintptr_t)f, bits);
|
|
|
|
CHECK_INITED;
|
|
|
|
gpioGetSamples.ex = 1;
|
|
gpioGetSamples.userdata = userdata;
|
|
gpioGetSamples.func = f;
|
|
|
|
if (f) gpioGetSamples.bits = bits;
|
|
else gpioGetSamples.bits = 0;
|
|
|
|
monitorBits = alertBits | notifyBits | scriptBits | gpioGetSamples.bits;
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
static int intGpioSetTimerFunc(unsigned id,
|
|
unsigned millis,
|
|
void *f,
|
|
int user,
|
|
void *userdata)
|
|
{
|
|
pthread_attr_t pthAttr;
|
|
|
|
DBG(DBG_INTERNAL, "id=%d millis=%d function=%08"PRIXPTR" user=%d userdata=%08"PRIXPTR,
|
|
id, millis, (uintptr_t)f, user, (uintptr_t)userdata);
|
|
|
|
gpioTimer[id].id = id;
|
|
|
|
if (f)
|
|
{
|
|
gpioTimer[id].func = f;
|
|
gpioTimer[id].ex = user;
|
|
gpioTimer[id].userdata = userdata;
|
|
gpioTimer[id].millis = millis;
|
|
|
|
if (!gpioTimer[id].running)
|
|
{
|
|
if (pthread_attr_init(&pthAttr))
|
|
SOFT_ERROR(PI_TIMER_FAILED,
|
|
"pthread_attr_init failed (%m)");
|
|
|
|
if (pthread_attr_setstacksize(&pthAttr, STACK_SIZE))
|
|
SOFT_ERROR(PI_TIMER_FAILED,
|
|
"pthread_attr_setstacksize failed (%m)");
|
|
|
|
if (pthread_create(
|
|
&gpioTimer[id].pthId, &pthAttr, pthTimerTick, &gpioTimer[id]))
|
|
SOFT_ERROR(PI_TIMER_FAILED,
|
|
"timer %d, create failed (%m)", id);
|
|
|
|
gpioTimer[id].running = 1;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (gpioTimer[id].running)
|
|
{
|
|
|
|
/* destroy thread */
|
|
|
|
if (pthread_self() == gpioTimer[id].pthId)
|
|
{
|
|
gpioTimer[id].running = 0;
|
|
gpioTimer[id].func = 0;
|
|
pthread_exit(NULL);
|
|
}
|
|
else
|
|
{
|
|
if (pthread_cancel(gpioTimer[id].pthId))
|
|
SOFT_ERROR(PI_TIMER_FAILED, "timer %d, cancel failed (%m)", id);
|
|
|
|
if (pthread_join(gpioTimer[id].pthId, NULL))
|
|
SOFT_ERROR(PI_TIMER_FAILED, "timer %d, join failed (%m)", id);
|
|
|
|
gpioTimer[id].running = 0;
|
|
gpioTimer[id].func = 0;
|
|
}
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioSetTimerFunc(unsigned id, unsigned millis, gpioTimerFunc_t f)
|
|
{
|
|
DBG(DBG_USER, "id=%d millis=%d function=%08"PRIXPTR, id, millis, (uintptr_t)f);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (id > PI_MAX_TIMER)
|
|
SOFT_ERROR(PI_BAD_TIMER, "bad timer id (%d)", id);
|
|
|
|
if (f)
|
|
{
|
|
if ((millis < PI_MIN_MS) || (millis > PI_MAX_MS))
|
|
SOFT_ERROR(PI_BAD_MS, "timer %d, bad millis (%d)", id, millis);
|
|
}
|
|
|
|
intGpioSetTimerFunc(id, millis, f, 0, NULL);
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioSetTimerFuncEx(unsigned id, unsigned millis, gpioTimerFuncEx_t f,
|
|
void * userdata)
|
|
{
|
|
DBG(DBG_USER, "id=%d millis=%d function=%08"PRIXPTR", userdata=%08"PRIXPTR,
|
|
id, millis, (uintptr_t)f, (uintptr_t)userdata);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (id > PI_MAX_TIMER)
|
|
SOFT_ERROR(PI_BAD_TIMER, "bad timer id (%d)", id);
|
|
|
|
if ((millis < PI_MIN_MS) || (millis > PI_MAX_MS))
|
|
SOFT_ERROR(PI_BAD_MS, "timer %d, bad millis (%d)", id, millis);
|
|
|
|
intGpioSetTimerFunc(id, millis, f, 1, userdata);
|
|
|
|
return 0;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
pthread_t *gpioStartThread(gpioThreadFunc_t f, void *userdata)
|
|
{
|
|
pthread_t *pth;
|
|
pthread_attr_t pthAttr;
|
|
|
|
DBG(DBG_USER, "f=%08"PRIXPTR", userdata=%08"PRIXPTR, (uintptr_t)f, (uintptr_t)userdata);
|
|
|
|
CHECK_INITED_RET_NULL_PTR;
|
|
|
|
pth = malloc(sizeof(pthread_t));
|
|
|
|
if (pth)
|
|
{
|
|
if (pthread_attr_init(&pthAttr))
|
|
{
|
|
free(pth);
|
|
SOFT_ERROR(NULL, "pthread_attr_init failed");
|
|
}
|
|
|
|
if (pthread_attr_setstacksize(&pthAttr, STACK_SIZE))
|
|
{
|
|
free(pth);
|
|
SOFT_ERROR(NULL, "pthread_attr_setstacksize failed");
|
|
}
|
|
|
|
if (pthread_create(pth, &pthAttr, f, userdata))
|
|
{
|
|
free(pth);
|
|
SOFT_ERROR(NULL, "pthread_create failed");
|
|
}
|
|
}
|
|
return pth;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
void gpioStopThread(pthread_t *pth)
|
|
{
|
|
DBG(DBG_USER, "pth=%08"PRIXPTR, (uintptr_t)pth);
|
|
|
|
CHECK_INITED_RET_NIL;
|
|
|
|
if (pth)
|
|
{
|
|
if (pthread_self() == *pth)
|
|
{
|
|
free(pth);
|
|
pthread_exit(NULL);
|
|
}
|
|
else
|
|
{
|
|
pthread_cancel(*pth);
|
|
pthread_join(*pth, NULL);
|
|
free(pth);
|
|
}
|
|
}
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioStoreScript(char *script)
|
|
{
|
|
static pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER;
|
|
gpioScript_t *s;
|
|
int status, slot, i;
|
|
|
|
DBG(DBG_USER, "script=[%s]", script);
|
|
|
|
CHECK_INITED;
|
|
|
|
slot = -1;
|
|
|
|
pthread_mutex_lock(&mutex);
|
|
|
|
for (i=0; i<PI_MAX_SCRIPTS; i++)
|
|
{
|
|
if (gpioScript[i].state == PI_SCRIPT_FREE)
|
|
{
|
|
slot = i;
|
|
gpioScript[slot].state = PI_SCRIPT_RESERVED;
|
|
break;
|
|
}
|
|
}
|
|
|
|
pthread_mutex_unlock(&mutex);
|
|
|
|
if (slot < 0) SOFT_ERROR(PI_NO_SCRIPT_ROOM, "no room for scripts");
|
|
|
|
s = &gpioScript[slot];
|
|
|
|
status = cmdParseScript(script, &s->script, 0);
|
|
|
|
if (status == 0)
|
|
{
|
|
s->request = PI_SCRIPT_HALT;
|
|
s->run_state = PI_SCRIPT_INITING;
|
|
|
|
pthread_cond_init(&s->pthCond, NULL);
|
|
pthread_mutex_init(&s->pthMutex, NULL);
|
|
|
|
s->id = slot;
|
|
|
|
gpioScript[slot].state = PI_SCRIPT_IN_USE;
|
|
|
|
s->pthIdp = gpioStartThread(pthScript, s);
|
|
|
|
status = slot;
|
|
}
|
|
else
|
|
{
|
|
if (s->script.par) free(s->script.par);
|
|
s->script.par = NULL;
|
|
gpioScript[slot].state = PI_SCRIPT_FREE;
|
|
}
|
|
|
|
return status;
|
|
}
|
|
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioRunScript(unsigned script_id, unsigned numParam, uint32_t *param)
|
|
{
|
|
int status = 0;
|
|
|
|
DBG(DBG_USER, "script_id=%d numParam=%d param=%08"PRIXPTR,
|
|
script_id, numParam, (uintptr_t)param);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (script_id >= PI_MAX_SCRIPTS)
|
|
SOFT_ERROR(PI_BAD_SCRIPT_ID, "bad script id(%d)", script_id);
|
|
|
|
if (numParam > PI_MAX_SCRIPT_PARAMS)
|
|
SOFT_ERROR(PI_TOO_MANY_PARAM, "bad number of parameters(%d)", numParam);
|
|
|
|
if (gpioScript[script_id].state == PI_SCRIPT_IN_USE)
|
|
{
|
|
pthread_mutex_lock(&gpioScript[script_id].pthMutex);
|
|
|
|
if (gpioScript[script_id].run_state != PI_SCRIPT_INITING)
|
|
{
|
|
if ((numParam > 0) && (param != 0))
|
|
{
|
|
memcpy(gpioScript[script_id].script.par, param,
|
|
sizeof(uint32_t) * numParam);
|
|
}
|
|
|
|
gpioScript[script_id].request = PI_SCRIPT_RUN;
|
|
|
|
pthread_cond_signal(&gpioScript[script_id].pthCond);
|
|
}
|
|
else
|
|
{
|
|
status = PI_SCRIPT_NOT_READY;
|
|
}
|
|
|
|
pthread_mutex_unlock(&gpioScript[script_id].pthMutex);
|
|
|
|
return status;
|
|
}
|
|
else
|
|
{
|
|
return PI_BAD_SCRIPT_ID;
|
|
}
|
|
}
|
|
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioUpdateScript(unsigned script_id, unsigned numParam, uint32_t *param)
|
|
{
|
|
DBG(DBG_USER, "script_id=%d numParam=%d param=%08"PRIXPTR,
|
|
script_id, numParam, (uintptr_t)param);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (script_id >= PI_MAX_SCRIPTS)
|
|
SOFT_ERROR(PI_BAD_SCRIPT_ID, "bad script id(%d)", script_id);
|
|
|
|
if (numParam > PI_MAX_SCRIPT_PARAMS)
|
|
SOFT_ERROR(PI_TOO_MANY_PARAM, "bad number of parameters(%d)", numParam);
|
|
|
|
if (gpioScript[script_id].state == PI_SCRIPT_IN_USE)
|
|
{
|
|
if ((numParam > 0) && (param != 0))
|
|
{
|
|
memcpy(gpioScript[script_id].script.par, param,
|
|
sizeof(uint32_t) * numParam);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
return PI_BAD_SCRIPT_ID;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioScriptStatus(unsigned script_id, uint32_t *param)
|
|
{
|
|
DBG(DBG_USER, "script_id=%d param=%08"PRIXPTR, script_id, (uintptr_t)param);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (script_id >= PI_MAX_SCRIPTS)
|
|
SOFT_ERROR(PI_BAD_SCRIPT_ID, "bad script id(%d)", script_id);
|
|
|
|
if (gpioScript[script_id].state == PI_SCRIPT_IN_USE)
|
|
{
|
|
if (param != NULL)
|
|
{
|
|
memcpy(param, gpioScript[script_id].script.par,
|
|
sizeof(uint32_t) * PI_MAX_SCRIPT_PARAMS);
|
|
}
|
|
|
|
return gpioScript[script_id].run_state;
|
|
}
|
|
else return PI_BAD_SCRIPT_ID;
|
|
}
|
|
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioStopScript(unsigned script_id)
|
|
{
|
|
DBG(DBG_USER, "script_id=%d", script_id);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (script_id >= PI_MAX_SCRIPTS)
|
|
SOFT_ERROR(PI_BAD_SCRIPT_ID, "bad script id(%d)", script_id);
|
|
|
|
if (gpioScript[script_id].state == PI_SCRIPT_IN_USE)
|
|
{
|
|
pthread_mutex_lock(&gpioScript[script_id].pthMutex);
|
|
|
|
gpioScript[script_id].request = PI_SCRIPT_HALT;
|
|
|
|
if (gpioScript[script_id].run_state == PI_SCRIPT_WAITING)
|
|
{
|
|
pthread_cond_signal(&gpioScript[script_id].pthCond);
|
|
}
|
|
|
|
pthread_mutex_unlock(&gpioScript[script_id].pthMutex);
|
|
|
|
return 0;
|
|
}
|
|
else return PI_BAD_SCRIPT_ID;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioDeleteScript(unsigned script_id)
|
|
{
|
|
DBG(DBG_USER, "script_id=%d", script_id);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (script_id >= PI_MAX_SCRIPTS)
|
|
SOFT_ERROR(PI_BAD_SCRIPT_ID, "bad script id(%d)", script_id);
|
|
|
|
if (gpioScript[script_id].state == PI_SCRIPT_IN_USE)
|
|
{
|
|
gpioScript[script_id].state = PI_SCRIPT_DYING;
|
|
|
|
pthread_mutex_lock(&gpioScript[script_id].pthMutex);
|
|
|
|
gpioScript[script_id].request = PI_SCRIPT_HALT;
|
|
|
|
if (gpioScript[script_id].run_state == PI_SCRIPT_WAITING)
|
|
{
|
|
pthread_cond_signal(&gpioScript[script_id].pthCond);
|
|
}
|
|
|
|
pthread_mutex_unlock(&gpioScript[script_id].pthMutex);
|
|
|
|
while (gpioScript[script_id].run_state == PI_SCRIPT_RUNNING)
|
|
{
|
|
myGpioSleep(0, 5000); /* give script time to halt */
|
|
}
|
|
|
|
gpioStopThread(gpioScript[script_id].pthIdp);
|
|
|
|
if (gpioScript[script_id].script.par)
|
|
free(gpioScript[script_id].script.par);
|
|
|
|
gpioScript[script_id].script.par = NULL;
|
|
|
|
gpioScript[script_id].state = PI_SCRIPT_FREE;
|
|
|
|
return 0;
|
|
}
|
|
else return PI_BAD_SCRIPT_ID;
|
|
}
|
|
|
|
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioSetSignalFunc(unsigned signum, gpioSignalFunc_t f)
|
|
{
|
|
DBG(DBG_USER, "signum=%d function=%08"PRIXPTR, signum, (uintptr_t)f);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (signum > PI_MAX_SIGNUM)
|
|
SOFT_ERROR(PI_BAD_SIGNUM, "bad signum (%d)", signum);
|
|
|
|
gpioSignal[signum].ex = 0;
|
|
gpioSignal[signum].userdata = NULL;
|
|
|
|
gpioSignal[signum].func = f;
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioSetSignalFuncEx(unsigned signum, gpioSignalFuncEx_t f,
|
|
void *userdata)
|
|
{
|
|
DBG(DBG_USER, "signum=%d function=%08"PRIXPTR" userdata=%08"PRIXPTR,
|
|
signum, (uintptr_t)f, (uintptr_t)userdata);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (signum > PI_MAX_SIGNUM)
|
|
SOFT_ERROR(PI_BAD_SIGNUM, "bad signum (%d)", signum);
|
|
|
|
gpioSignal[signum].ex = 1;
|
|
gpioSignal[signum].userdata = userdata;
|
|
|
|
gpioSignal[signum].func = f;
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
uint32_t gpioRead_Bits_0_31(void)
|
|
{
|
|
DBG(DBG_USER, "");
|
|
|
|
CHECK_INITED;
|
|
|
|
return (*(gpioReg + GPLEV0));
|
|
}
|
|
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
uint32_t gpioRead_Bits_32_53(void)
|
|
{
|
|
DBG(DBG_USER, "");
|
|
|
|
CHECK_INITED;
|
|
|
|
return (*(gpioReg + GPLEV1));
|
|
}
|
|
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioWrite_Bits_0_31_Clear(uint32_t bits)
|
|
{
|
|
DBG(DBG_USER, "bits=%08X", bits);
|
|
|
|
CHECK_INITED;
|
|
|
|
*(gpioReg + GPCLR0) = bits;
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioWrite_Bits_32_53_Clear(uint32_t bits)
|
|
{
|
|
DBG(DBG_USER, "bits=%08X", bits);
|
|
|
|
CHECK_INITED;
|
|
|
|
*(gpioReg + GPCLR1) = bits;
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioWrite_Bits_0_31_Set(uint32_t bits)
|
|
{
|
|
DBG(DBG_USER, "bits=%08X", bits);
|
|
|
|
CHECK_INITED;
|
|
|
|
*(gpioReg + GPSET0) = bits;
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioWrite_Bits_32_53_Set(uint32_t bits)
|
|
{
|
|
DBG(DBG_USER, "bits=%08X", bits);
|
|
|
|
CHECK_INITED;
|
|
|
|
*(gpioReg + GPSET1) = bits;
|
|
|
|
return 0;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioHardwareClock(unsigned gpio, unsigned frequency)
|
|
{
|
|
int cctl[] = {CLK_GP0_CTL, CLK_GP1_CTL, CLK_GP2_CTL};
|
|
int cdiv[] = {CLK_GP0_DIV, CLK_GP1_DIV, CLK_GP2_DIV};
|
|
int csrc[CLK_SRCS] = {CLK_CTL_SRC_OSC, CLK_CTL_SRC_PLLD};
|
|
uint32_t cfreq[CLK_SRCS]={clk_osc_freq, clk_plld_freq};
|
|
unsigned clock, mode, mash;
|
|
int password = 0;
|
|
double f;
|
|
clkInf_t clkInf={0,0,0};
|
|
|
|
DBG(DBG_USER, "gpio=%d frequency=%d", gpio, frequency);
|
|
|
|
CHECK_INITED;
|
|
|
|
if ((gpio >> 24) == 0x5A) password = 1;
|
|
|
|
gpio &= 0xFFFFFF;
|
|
|
|
if (gpio > PI_MAX_GPIO)
|
|
SOFT_ERROR(PI_BAD_GPIO, "bad gpio (%d)", gpio);
|
|
|
|
if (!clkDef[gpio])
|
|
SOFT_ERROR(PI_NOT_HCLK_GPIO, "bad gpio for clock (%d)", gpio);
|
|
|
|
if (((frequency < hw_clk_min_freq) ||
|
|
(frequency > hw_clk_max_freq)) &&
|
|
(frequency))
|
|
SOFT_ERROR(PI_BAD_HCLK_FREQ,
|
|
"bad hardware clock frequency %d-%d: (%d)",
|
|
hw_clk_min_freq, hw_clk_max_freq, frequency);
|
|
|
|
clock = (clkDef[gpio] >> 4) & 3;
|
|
|
|
if ((clock == 1) && (!password))
|
|
SOFT_ERROR(PI_BAD_HCLK_PASS,
|
|
"Need password to use clock 1 (%d)", gpio);
|
|
|
|
mode = clkDef[gpio] & 7;
|
|
mash = frequency < PI_MASH_MAX_FREQ ? 1 : 0;
|
|
|
|
if (frequency)
|
|
{
|
|
if (chooseBestClock(&clkInf, frequency, CLK_SRCS, cfreq))
|
|
{
|
|
if (clkInf.frac == 0) mash = 0;
|
|
|
|
initHWClk(cctl[clock], cdiv[clock],
|
|
csrc[clkInf.clock], clkInf.div, clkInf.frac, mash);
|
|
|
|
myGpioSetMode(gpio, mode);
|
|
|
|
gpioInfo[gpio].is = GPIO_HW_CLK;
|
|
|
|
f = (double) cfreq[clkInf.clock] /
|
|
((double)clkInf.div + ((double)clkInf.frac / 4096.0));
|
|
|
|
hw_clk_freq[clock] = (f + 0.5);
|
|
|
|
DBG(DBG_USER, "cf=%d div=%d frac=%d mash=%d",
|
|
cfreq[clkInf.clock], clkInf.div, clkInf.frac, mash);
|
|
}
|
|
else
|
|
{
|
|
SOFT_ERROR(PI_BAD_HCLK_FREQ,
|
|
"bad hardware clock frequency %d-%d: (%d)",
|
|
hw_clk_min_freq, hw_clk_max_freq, frequency);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
/* frequency 0, stop clock */
|
|
clkReg[cctl[clock]] = BCM_PASSWD | CLK_CTL_KILL;
|
|
|
|
if (gpioInfo[gpio].is == GPIO_HW_CLK)
|
|
gpioInfo[gpio].is = GPIO_UNDEFINED;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioHardwarePWM(
|
|
unsigned gpio, unsigned frequency, unsigned dutycycle)
|
|
{
|
|
uint32_t old_PWM_CTL;
|
|
unsigned pwm, mode;
|
|
uint32_t real_range, real_dutycycle;
|
|
|
|
DBG(DBG_USER, "gpio=%d frequency=%d dutycycle=%d",
|
|
gpio, frequency, dutycycle);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (gpio > PI_MAX_GPIO)
|
|
SOFT_ERROR(PI_BAD_GPIO, "bad gpio (%d)", gpio);
|
|
|
|
if (!PWMDef[gpio])
|
|
SOFT_ERROR(PI_NOT_HPWM_GPIO, "bad gpio for PWM (%d)", gpio);
|
|
|
|
if (dutycycle > PI_HW_PWM_RANGE)
|
|
SOFT_ERROR(PI_BAD_HPWM_DUTY, "bad PWM dutycycle (%d)", dutycycle);
|
|
|
|
if (((frequency < PI_HW_PWM_MIN_FREQ) ||
|
|
(frequency > hw_pwm_max_freq)) &&
|
|
(frequency))
|
|
SOFT_ERROR(PI_BAD_HPWM_FREQ,
|
|
"bad hardware PWM frequency %d-%d: (%d)",
|
|
PI_HW_PWM_MIN_FREQ, hw_pwm_max_freq, frequency);
|
|
|
|
|
|
if (gpioCfg.clockPeriph == PI_CLOCK_PWM)
|
|
SOFT_ERROR(PI_HPWM_ILLEGAL, "illegal, PWM in use for main clock");
|
|
|
|
pwm = (PWMDef[gpio] >> 4) & 3;
|
|
mode = PWMDef[gpio] & 7;
|
|
|
|
if (frequency)
|
|
{
|
|
real_range = ((double)clk_plld_freq / (2.0 * frequency)) + 0.5;
|
|
real_dutycycle = ((uint64_t)dutycycle * real_range) / PI_HW_PWM_RANGE;
|
|
|
|
/* record the set PWM frequency and dutycycle */
|
|
|
|
hw_pwm_freq[pwm] =
|
|
((double)clk_plld_freq / ( 2.0 * real_range)) + 0.5;
|
|
|
|
hw_pwm_duty[pwm] = dutycycle;
|
|
|
|
hw_pwm_real_range[pwm] = real_range;
|
|
|
|
/* Abort any waveform transmission in progress */
|
|
|
|
if (gpioWaveTxBusy()) gpioWaveTxStop();
|
|
|
|
waveClockInited = 0;
|
|
|
|
/* preserve channel enable only and mark space mode */
|
|
|
|
old_PWM_CTL = pwmReg[PWM_CTL] &
|
|
(PWM_CTL_PWEN1 | PWM_CTL_MSEN1 | PWM_CTL_PWEN2 | PWM_CTL_MSEN2);
|
|
|
|
if (!PWMClockInited)
|
|
{
|
|
pwmReg[PWM_CTL] = 0;
|
|
|
|
myGpioDelay(10);
|
|
|
|
initHWClk(CLK_PWMCTL, CLK_PWMDIV, CLK_CTL_SRC_PLLD, 2, 0, 0);
|
|
|
|
PWMClockInited = 1;
|
|
}
|
|
|
|
if (pwm == 0)
|
|
{
|
|
pwmReg[PWM_RNG1] = real_range;
|
|
myGpioDelay(10);
|
|
pwmReg[PWM_DAT1] = real_dutycycle;
|
|
myGpioDelay(10);
|
|
|
|
pwmReg[PWM_CTL] = (old_PWM_CTL | PWM_CTL_PWEN1 | PWM_CTL_MSEN1);
|
|
}
|
|
else
|
|
{
|
|
pwmReg[PWM_RNG2] = real_range;
|
|
myGpioDelay(10);
|
|
pwmReg[PWM_DAT2] = real_dutycycle;
|
|
myGpioDelay(10);
|
|
|
|
pwmReg[PWM_CTL] = (old_PWM_CTL | PWM_CTL_PWEN2 | PWM_CTL_MSEN2);
|
|
}
|
|
|
|
if (gpioInfo[gpio].is != GPIO_HW_PWM)
|
|
{
|
|
switchFunctionOff(gpio);
|
|
|
|
myGpioSetMode(gpio, mode);
|
|
|
|
gpioInfo[gpio].is = GPIO_HW_PWM;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
/* frequency 0, stop PWM */
|
|
|
|
if (gpioInfo[gpio].is == GPIO_HW_PWM)
|
|
{
|
|
if (pwm == 0) pwmReg[PWM_CTL] &= (~PWM_CTL_PWEN1);
|
|
else pwmReg[PWM_CTL] &= (~PWM_CTL_PWEN2);
|
|
|
|
gpioInfo[gpio].is = GPIO_UNDEFINED;
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
int gpioSetPad(unsigned pad, unsigned padStrength)
|
|
{
|
|
DBG(DBG_USER, "pad=%d padStrength=%d", pad, padStrength);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (pad > PI_MAX_PAD)
|
|
SOFT_ERROR(PI_BAD_PAD, "bad pad number (%d)", pad);
|
|
|
|
if ((padStrength < PI_MIN_PAD_STRENGTH) ||
|
|
(padStrength > PI_MAX_PAD_STRENGTH))
|
|
SOFT_ERROR(PI_BAD_STRENGTH, "bad pad drive strength (%d)", pad);
|
|
|
|
/* 1-16 -> 0-7 */
|
|
|
|
padStrength += 1;
|
|
padStrength /= 2;
|
|
padStrength -= 1;
|
|
|
|
padsReg[11+pad] = BCM_PASSWD | 0x18 | (padStrength & 7) ;
|
|
|
|
return 0;
|
|
}
|
|
|
|
int gpioGetPad(unsigned pad)
|
|
{
|
|
int strength;
|
|
|
|
DBG(DBG_USER, "pad=%d", pad);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (pad > PI_MAX_PAD)
|
|
SOFT_ERROR(PI_BAD_PAD, "bad pad (%d)", pad);
|
|
|
|
strength = padsReg[11+pad] & 7;
|
|
|
|
strength *= 2;
|
|
strength += 2;
|
|
|
|
return strength;
|
|
}
|
|
|
|
int shell(char *scriptName, char *scriptString)
|
|
{
|
|
int status;
|
|
char buf[4096];
|
|
|
|
DBG(DBG_USER, "name=%s string=%s", scriptName, scriptString);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (!myScriptNameValid(scriptName))
|
|
SOFT_ERROR(PI_BAD_SCRIPT_NAME, "bad script name (%s)", scriptName);
|
|
|
|
snprintf(buf, sizeof(buf),
|
|
"/opt/pigpio/cgi/%s %s", scriptName, scriptString);
|
|
|
|
DBG(DBG_USER, "%s", buf);
|
|
|
|
status = system(buf);
|
|
|
|
if (status < 0) status = PI_BAD_SHELL_STATUS;
|
|
|
|
return status;
|
|
}
|
|
|
|
|
|
int fileApprove(char *filename)
|
|
{
|
|
char match[PI_MAX_PATH];
|
|
char buffer[PI_MAX_PATH];
|
|
char line[PI_MAX_PATH];
|
|
char mperm=0;
|
|
char perm;
|
|
char term;
|
|
FILE *f;
|
|
|
|
buffer[0] = 0;
|
|
match[0] = 0;
|
|
|
|
if (myPathBad(filename)) return PI_FILE_NONE;
|
|
|
|
f = fopen("/opt/pigpio/access", "r");
|
|
|
|
if (!f) return PI_FILE_NONE;
|
|
|
|
while (!feof(f))
|
|
{
|
|
buffer[0] = 0;
|
|
perm = 0;
|
|
term = 0;
|
|
if (fgets(line, sizeof(line), f))
|
|
{
|
|
sscanf(line, " %511s %c%c", buffer, &perm, &term);
|
|
if (term == 10)
|
|
{
|
|
if (myPathBad(buffer)) continue; /* disallow risky lines */
|
|
|
|
if (fnmatch(buffer, filename, 0) == 0)
|
|
{
|
|
if (match[0])
|
|
{
|
|
if (fnmatch(match, buffer, 0) == 0)
|
|
{
|
|
strcpy(match, buffer);
|
|
mperm = perm;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
strcpy(match, buffer);
|
|
mperm = perm;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
fclose(f);
|
|
|
|
if (match[0])
|
|
{
|
|
switch (toupper(mperm))
|
|
{
|
|
case 'R': return PI_FILE_READ;
|
|
case 'W': return PI_FILE_WRITE;
|
|
case 'U': return PI_FILE_RW;
|
|
default : return PI_FILE_NONE;
|
|
}
|
|
}
|
|
|
|
return PI_FILE_NONE;
|
|
}
|
|
|
|
int fileOpen(char *file, unsigned mode)
|
|
{
|
|
static pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER;
|
|
int fd=-1;
|
|
int i, slot, oflag, omode, pmode, rmode;
|
|
struct stat statbuf;
|
|
|
|
DBG(DBG_USER, "file=%s mode=%d", file, mode);
|
|
|
|
CHECK_INITED;
|
|
|
|
if ( (mode < PI_FILE_MIN) ||
|
|
(mode > PI_FILE_MAX) ||
|
|
((mode & PI_FILE_RW) == 0) )
|
|
SOFT_ERROR(PI_BAD_FILE_MODE, "bad mode (%d)", mode);
|
|
|
|
pmode = fileApprove(file); // 0=NONE, 1=READ, 2=WRITE, 3=RW
|
|
rmode = mode & PI_FILE_RW; // 0=NONE, 1=READ, 2=WRITE, 3=RW
|
|
|
|
if (((pmode & rmode) != rmode) || (rmode == PI_FILE_NONE))
|
|
SOFT_ERROR(PI_NO_FILE_ACCESS, "no permission to access file (%s)", file);
|
|
|
|
if ((mode > 3) && ((mode & PI_FILE_WRITE) == 0))
|
|
SOFT_ERROR(PI_NO_FILE_ACCESS, "no permission to write file (%s)", file);
|
|
|
|
slot = -1;
|
|
|
|
pthread_mutex_lock(&mutex);
|
|
|
|
for (i=0; i<PI_FILE_SLOTS; i++)
|
|
{
|
|
if (fileInfo[i].state == PI_FILE_CLOSED)
|
|
{
|
|
slot = i;
|
|
fileInfo[slot].state = PI_FILE_RESERVED;
|
|
break;
|
|
}
|
|
}
|
|
|
|
pthread_mutex_unlock(&mutex);
|
|
|
|
if (slot < 0) SOFT_ERROR(PI_NO_HANDLE, "no file handles");
|
|
|
|
omode = 0;
|
|
oflag = 0;
|
|
|
|
if (mode & PI_FILE_APPEND)
|
|
{
|
|
oflag |= O_APPEND;
|
|
}
|
|
|
|
if (mode & PI_FILE_CREATE)
|
|
{
|
|
oflag |= O_CREAT;
|
|
omode |= (S_IRUSR|S_IWUSR);
|
|
}
|
|
|
|
if (mode & PI_FILE_TRUNC)
|
|
{
|
|
oflag |= O_TRUNC;
|
|
}
|
|
|
|
switch(mode&PI_FILE_RW)
|
|
{
|
|
case PI_FILE_READ:
|
|
fd = open(file, O_RDONLY|oflag, omode);
|
|
break;
|
|
|
|
case PI_FILE_WRITE:
|
|
fd = open(file, O_WRONLY|oflag, omode);
|
|
break;
|
|
|
|
case PI_FILE_RW:
|
|
fd = open(file, O_RDWR|oflag, omode);
|
|
break;
|
|
}
|
|
|
|
if (fd == -1)
|
|
{
|
|
fileInfo[slot].state = PI_FILE_CLOSED;
|
|
return PI_FIL_OPEN_FAILED;
|
|
}
|
|
else
|
|
{
|
|
if (stat(file, &statbuf) == 0)
|
|
{
|
|
if (S_ISDIR(statbuf.st_mode))
|
|
{
|
|
close(fd);
|
|
fileInfo[slot].state = PI_FILE_CLOSED;
|
|
SOFT_ERROR(PI_FILE_IS_A_DIR, "file is a directory (%s)", file);
|
|
}
|
|
}
|
|
}
|
|
|
|
fileInfo[slot].fd = fd;
|
|
fileInfo[slot].mode = mode;
|
|
fileInfo[slot].state = PI_FILE_OPENED;
|
|
|
|
return slot;
|
|
}
|
|
|
|
int fileClose(unsigned handle)
|
|
{
|
|
DBG(DBG_USER, "handle=%d", handle);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (handle >= PI_FILE_SLOTS)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
if (fileInfo[handle].state != PI_FILE_OPENED)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
if (fileInfo[handle].fd >= 0) close(fileInfo[handle].fd);
|
|
|
|
fileInfo[handle].fd = -1;
|
|
fileInfo[handle].state = PI_FILE_CLOSED;
|
|
|
|
return 0;
|
|
}
|
|
|
|
int fileWrite(unsigned handle, char *buf, unsigned count)
|
|
{
|
|
int w;
|
|
|
|
DBG(DBG_USER, "handle=%d count=%d [%s]",
|
|
handle, count, myBuf2Str(count, buf));
|
|
|
|
CHECK_INITED;
|
|
|
|
if (handle >= PI_FILE_SLOTS)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
if (fileInfo[handle].state != PI_FILE_OPENED)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
if (!count)
|
|
SOFT_ERROR(PI_BAD_PARAM, "bad count (%d)", count);
|
|
|
|
if (!(fileInfo[handle].mode & PI_FILE_WRITE))
|
|
SOFT_ERROR(PI_FILE_NOT_WOPEN, "file not opened for write");
|
|
|
|
w = write(fileInfo[handle].fd, buf, count);
|
|
|
|
if (w != count)
|
|
{
|
|
if (w == -1) DBG(DBG_USER, "write failed with errno %d", errno);
|
|
|
|
return PI_BAD_FILE_WRITE;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
int fileRead(unsigned handle, char *buf, unsigned count)
|
|
{
|
|
int r;
|
|
|
|
DBG(DBG_USER, "handle=%d count=%d buf=0x%"PRIXPTR, handle, count, (uintptr_t)buf);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (handle >= PI_FILE_SLOTS)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
if (fileInfo[handle].state != PI_FILE_OPENED)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
if (!count)
|
|
SOFT_ERROR(PI_BAD_PARAM, "bad count (%d)", count);
|
|
|
|
if (!(fileInfo[handle].mode & PI_FILE_READ))
|
|
SOFT_ERROR(PI_FILE_NOT_ROPEN, "file not opened for read");
|
|
|
|
r = read(fileInfo[handle].fd, buf, count);
|
|
|
|
if (r == -1)
|
|
{
|
|
DBG(DBG_USER, "read failed with errno %d", errno);
|
|
return PI_BAD_FILE_READ;
|
|
}
|
|
else
|
|
{
|
|
buf[r] = 0;
|
|
return r;
|
|
}
|
|
}
|
|
|
|
|
|
int fileSeek(unsigned handle, int32_t seekOffset, int seekFrom)
|
|
{
|
|
int whence, s;
|
|
|
|
DBG(DBG_USER, "handle=%d offset=%d from=%d",
|
|
handle, seekOffset, seekFrom);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (handle >= PI_FILE_SLOTS)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
if (fileInfo[handle].state != PI_FILE_OPENED)
|
|
SOFT_ERROR(PI_BAD_HANDLE, "bad handle (%d)", handle);
|
|
|
|
switch (seekFrom)
|
|
{
|
|
case PI_FROM_START:
|
|
whence = SEEK_SET;
|
|
break;
|
|
|
|
case PI_FROM_CURRENT:
|
|
whence = SEEK_CUR;
|
|
break;
|
|
|
|
case PI_FROM_END:
|
|
whence = SEEK_END;
|
|
break;
|
|
|
|
default:
|
|
SOFT_ERROR(PI_BAD_FILE_SEEK, "bad seek from (%d)", seekFrom);
|
|
}
|
|
|
|
s = lseek(fileInfo[handle].fd, seekOffset, whence);
|
|
|
|
if (s == -1)
|
|
{
|
|
DBG(DBG_USER, "seek failed with errno %d", errno);
|
|
return PI_BAD_FILE_SEEK;
|
|
}
|
|
|
|
return s;
|
|
}
|
|
|
|
int fileList(char *fpat, char *buf, unsigned count)
|
|
{
|
|
int len, bufpos;
|
|
glob_t pglob;
|
|
int i;
|
|
|
|
DBG(DBG_USER, "fpat=%s count=%d buf=%"PRIxPTR, fpat, count, (uintptr_t)buf);
|
|
|
|
CHECK_INITED;
|
|
|
|
if ((fileApprove(fpat) & PI_FILE_READ) != PI_FILE_READ)
|
|
SOFT_ERROR(PI_NO_FILE_ACCESS, "no permission to access file (%s)", fpat);
|
|
|
|
bufpos = 0;
|
|
|
|
if (glob(fpat, GLOB_MARK, NULL, &pglob) == 0)
|
|
{
|
|
for (i=0; i<pglob.gl_pathc; i++)
|
|
{
|
|
len = strlen(pglob.gl_pathv[i]);
|
|
if ((bufpos + len + 1) < count)
|
|
{
|
|
strcpy(buf+bufpos, pglob.gl_pathv[i]);
|
|
bufpos += len;
|
|
buf[bufpos++] = '\n';
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
bufpos = PI_NO_FILE_MATCH;
|
|
}
|
|
|
|
globfree(&pglob);
|
|
|
|
return bufpos;
|
|
}
|
|
|
|
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioTime(unsigned timetype, int *seconds, int *micros)
|
|
{
|
|
struct timespec ts;
|
|
|
|
DBG(DBG_USER, "timetype=%d &seconds=%08"PRIXPTR" µs=%08"PRIXPTR,
|
|
timetype, (uintptr_t)seconds, (uintptr_t)micros);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (timetype > PI_TIME_ABSOLUTE)
|
|
SOFT_ERROR(PI_BAD_TIMETYPE, "bad timetype (%d)", timetype);
|
|
|
|
if (timetype == PI_TIME_ABSOLUTE)
|
|
{
|
|
clock_gettime(CLOCK_REALTIME, &ts);
|
|
*seconds = ts.tv_sec;
|
|
*micros = ts.tv_nsec/1000;
|
|
}
|
|
else
|
|
{
|
|
clock_gettime(CLOCK_REALTIME, &ts);
|
|
|
|
TIMER_SUB(&ts, &libStarted, &ts);
|
|
|
|
*seconds = ts.tv_sec;
|
|
*micros = ts.tv_nsec/1000;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioSleep(unsigned timetype, int seconds, int micros)
|
|
{
|
|
struct timespec ts, rem;
|
|
|
|
DBG(DBG_USER, "timetype=%d seconds=%d micros=%d",
|
|
timetype, seconds, micros);
|
|
|
|
CHECK_INITED;
|
|
|
|
if (timetype > PI_TIME_ABSOLUTE)
|
|
SOFT_ERROR(PI_BAD_TIMETYPE, "bad timetype (%d)", timetype);
|
|
|
|
if (seconds < 0)
|
|
SOFT_ERROR(PI_BAD_SECONDS, "bad seconds (%d)", seconds);
|
|
|
|
if ((micros < 0) || (micros > 999999))
|
|
SOFT_ERROR(PI_BAD_MICROS, "bad micros (%d)", micros);
|
|
|
|
ts.tv_sec = seconds;
|
|
ts.tv_nsec = micros * 1000;
|
|
|
|
if (timetype == PI_TIME_ABSOLUTE)
|
|
{
|
|
while (clock_nanosleep(CLOCK_REALTIME, TIMER_ABSTIME, &ts, &rem));
|
|
}
|
|
else
|
|
{
|
|
while (clock_nanosleep(CLOCK_REALTIME, 0, &ts, &rem))
|
|
{
|
|
/* copy remaining time to ts */
|
|
ts.tv_sec = rem.tv_sec;
|
|
ts.tv_nsec = rem.tv_nsec;
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
uint32_t gpioDelay(uint32_t micros)
|
|
{
|
|
uint32_t start;
|
|
|
|
DBG(DBG_USER, "microseconds=%u", micros);
|
|
|
|
CHECK_INITED;
|
|
|
|
start = systReg[SYST_CLO];
|
|
|
|
if (micros <= PI_MAX_BUSY_DELAY)
|
|
while ((systReg[SYST_CLO] - start) <= micros);
|
|
else
|
|
gpioSleep(PI_TIME_RELATIVE, (micros/MILLION), (micros%MILLION));
|
|
|
|
return (systReg[SYST_CLO] - start);
|
|
}
|
|
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
uint32_t gpioTick(void)
|
|
{
|
|
CHECK_INITED;
|
|
|
|
return systReg[SYST_CLO];
|
|
}
|
|
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
unsigned gpioVersion(void)
|
|
{
|
|
DBG(DBG_USER, "");
|
|
|
|
return PIGPIO_VERSION;
|
|
}
|
|
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
/*
|
|
2 2 2 2 2 2 1 1 1 1 1 1 1 1 1 1 0 0 0 0 0 0 0 0 0 0
|
|
5 4 3 2 1 0 9 8 7 6 5 4 3 2 1 0 9 8 7 6 5 4 3 2 1 0
|
|
|
|
W W S M M M B B B B P P P P T T T T T T T T R R R R
|
|
|
|
W warranty void if either bit is set
|
|
|
|
S 0=old (bits 0-22 are revision number) 1=new (following fields apply)
|
|
|
|
M 0=256 1=512 2=1024 3=2GB 4=4GB
|
|
|
|
B 0=Sony 1=Egoman 2=Embest 3=Sony Japan 4=Embest 5=Stadium
|
|
|
|
P 0=2835, 1=2836, 2=2837 3=2711
|
|
|
|
T 0=A 1=B 2=A+ 3=B+ 4=Pi2B 5=Alpha 6=CM1 8=Pi3B 9=Zero a=CM3 c=Zero W
|
|
d=3B+ e=3A+ 10=CM3+ 11=4B
|
|
|
|
R PCB board revision
|
|
|
|
*/
|
|
|
|
unsigned gpioHardwareRevision(void)
|
|
{
|
|
static unsigned rev = 0;
|
|
|
|
FILE * filp;
|
|
char buf[512];
|
|
char term;
|
|
|
|
DBG(DBG_USER, "");
|
|
|
|
if (rev) return rev;
|
|
|
|
filp = fopen ("/proc/cpuinfo", "r");
|
|
|
|
|
|
if (filp != NULL)
|
|
{
|
|
while (fgets(buf, sizeof(buf), filp) != NULL)
|
|
{
|
|
if (!strncasecmp("revision\t:", buf, 10))
|
|
{
|
|
if (sscanf(buf+10, "%x%c", &rev, &term) == 2)
|
|
{
|
|
if (term != '\n') rev = 0;
|
|
}
|
|
}
|
|
}
|
|
fclose(filp);
|
|
}
|
|
|
|
/* (some) arm64 operating systems get revision number here */
|
|
|
|
if (rev == 0)
|
|
{
|
|
DBG(DBG_USER, "searching /proc/device-tree for revision");
|
|
filp = fopen ("/proc/device-tree/system/linux,revision", "r");
|
|
|
|
if (filp != NULL)
|
|
{
|
|
uint32_t tmp;
|
|
if (fread(&tmp,1 , 4, filp) == 4)
|
|
{
|
|
/*
|
|
for some reason the value returned by reading
|
|
this /proc entry seems to be big endian,
|
|
convert it.
|
|
*/
|
|
rev = ntohl(tmp);
|
|
rev &= 0xFFFFFF; /* mask out warranty bit */
|
|
}
|
|
fclose(filp);
|
|
}
|
|
}
|
|
|
|
piCores = 0;
|
|
pi_ispi = 0;
|
|
rev &= 0xFFFFFF; /* mask out warranty bit */
|
|
|
|
/* Decode revision code */
|
|
|
|
if ((rev & 0x800000) == 0) /* old rev code */
|
|
{
|
|
if ((rev > 0) && (rev < 0x0016)) /* all BCM2835 */
|
|
{
|
|
pi_ispi = 1;
|
|
piCores = 1;
|
|
pi_peri_phys = 0x20000000;
|
|
pi_dram_bus = 0x40000000;
|
|
pi_mem_flag = 0x0C;
|
|
}
|
|
else
|
|
{
|
|
DBG(DBG_ALWAYS, "unknown revision=%x", rev);
|
|
rev = 0;
|
|
}
|
|
}
|
|
else /* new rev code */
|
|
{
|
|
switch ((rev >> 12) & 0xF) /* just interested in BCM model */
|
|
{
|
|
|
|
case 0x0: /* BCM2835 */
|
|
pi_ispi = 1;
|
|
piCores = 1;
|
|
pi_peri_phys = 0x20000000;
|
|
pi_dram_bus = 0x40000000;
|
|
pi_mem_flag = 0x0C;
|
|
break;
|
|
|
|
case 0x1: /* BCM2836 */
|
|
case 0x2: /* BCM2837 */
|
|
pi_ispi = 1;
|
|
piCores = 4;
|
|
pi_peri_phys = 0x3F000000;
|
|
pi_dram_bus = 0xC0000000;
|
|
pi_mem_flag = 0x04;
|
|
break;
|
|
|
|
case 0x3: /* BCM2711 */
|
|
pi_ispi = 1;
|
|
piCores = 4;
|
|
pi_peri_phys = 0xFE000000;
|
|
pi_dram_bus = 0xC0000000;
|
|
pi_mem_flag = 0x04;
|
|
pi_is_2711 = 1;
|
|
clk_osc_freq = CLK_OSC_FREQ_2711;
|
|
clk_plld_freq = CLK_PLLD_FREQ_2711;
|
|
hw_pwm_max_freq = PI_HW_PWM_MAX_FREQ_2711;
|
|
hw_clk_min_freq = PI_HW_CLK_MIN_FREQ_2711;
|
|
hw_clk_max_freq = PI_HW_CLK_MAX_FREQ_2711;
|
|
break;
|
|
|
|
default:
|
|
DBG(DBG_ALWAYS, "unknown rev code (%x)", rev);
|
|
rev=0;
|
|
pi_ispi = 0;
|
|
break;
|
|
}
|
|
}
|
|
|
|
DBG(DBG_USER, "revision=%x", rev);
|
|
DBG(DBG_USER, "pi_peri_phys=%x", pi_peri_phys);
|
|
DBG(DBG_USER, "pi_dram_bus=%x", pi_dram_bus);
|
|
DBG(DBG_USER, "pi_mem_flag=%x", pi_mem_flag);
|
|
|
|
return rev;
|
|
}
|
|
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioCfgBufferSize(unsigned millis)
|
|
{
|
|
DBG(DBG_USER, "millis=%d", millis);
|
|
|
|
CHECK_NOT_INITED;
|
|
|
|
if ((millis < PI_BUF_MILLIS_MIN) || (millis > PI_BUF_MILLIS_MAX))
|
|
SOFT_ERROR(PI_BAD_BUF_MILLIS, "bad millis (%d)", millis);
|
|
|
|
gpioCfg.bufferMilliseconds = millis;
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioCfgClock(unsigned micros, unsigned peripheral, unsigned source)
|
|
{
|
|
DBG(DBG_USER, "micros=%d peripheral=%d", micros, peripheral);
|
|
|
|
CHECK_NOT_INITED;
|
|
|
|
if ((micros < 1) || (micros > 10))
|
|
SOFT_ERROR(PI_BAD_CLK_MICROS, "bad micros (%d)", micros);
|
|
|
|
if (!clkCfg[micros].valid)
|
|
SOFT_ERROR(PI_BAD_CLK_MICROS, "bad micros (%d)", micros);
|
|
|
|
if (peripheral > PI_CLOCK_PCM)
|
|
SOFT_ERROR(PI_BAD_CLK_PERIPH, "bad peripheral (%d)", peripheral);
|
|
|
|
gpioCfg.clockMicros = micros;
|
|
gpioCfg.clockPeriph = peripheral;
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioCfgDMAchannel(unsigned DMAchannel)
|
|
{
|
|
DBG(DBG_USER, "channel=%d", DMAchannel);
|
|
|
|
CHECK_NOT_INITED;
|
|
|
|
if ((DMAchannel < PI_MIN_DMA_CHANNEL) || (DMAchannel > PI_MAX_DMA_CHANNEL))
|
|
SOFT_ERROR(PI_BAD_CHANNEL, "bad channel (%d)", DMAchannel);
|
|
|
|
gpioCfg.DMAprimaryChannel = DMAchannel;
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioCfgDMAchannels(unsigned primaryChannel, unsigned secondaryChannel)
|
|
{
|
|
DBG(DBG_USER, "primary channel=%d, secondary channel=%d",
|
|
primaryChannel, secondaryChannel);
|
|
|
|
CHECK_NOT_INITED;
|
|
|
|
if (primaryChannel > PI_MAX_DMA_CHANNEL)
|
|
SOFT_ERROR(PI_BAD_PRIM_CHANNEL, "bad primary channel (%d)",
|
|
primaryChannel);
|
|
|
|
if ((secondaryChannel > PI_MAX_DMA_CHANNEL) ||
|
|
((secondaryChannel == primaryChannel) &&
|
|
(secondaryChannel != PI_DEFAULT_DMA_NOT_SET)))
|
|
SOFT_ERROR(PI_BAD_SECO_CHANNEL, "bad secondary channel (%d)",
|
|
secondaryChannel);
|
|
|
|
gpioCfg.DMAprimaryChannel = primaryChannel;
|
|
gpioCfg.DMAsecondaryChannel = secondaryChannel;
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
/*-------------------------------------------------------------------------*/
|
|
|
|
int gpioCfgPermissions(uint64_t updateMask)
|
|
{
|
|
DBG(DBG_USER, "gpio update mask=%"PRIX64, updateMask);
|
|
|
|
CHECK_NOT_INITED;
|
|
|
|
gpioMask = updateMask;
|
|
|
|
gpioMaskSet = 1;
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioCfgInterfaces(unsigned ifFlags)
|
|
{
|
|
DBG(DBG_USER, "ifFlags=%X", ifFlags);
|
|
|
|
CHECK_NOT_INITED;
|
|
|
|
if (ifFlags > 15)
|
|
SOFT_ERROR(PI_BAD_IF_FLAGS, "bad ifFlags (%X)", ifFlags);
|
|
|
|
gpioCfg.ifFlags = ifFlags;
|
|
|
|
return 0;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioCfgSocketPort(unsigned port)
|
|
{
|
|
DBG(DBG_USER, "port=%d", port);
|
|
|
|
CHECK_NOT_INITED;
|
|
|
|
if ((port < PI_MIN_SOCKET_PORT) || (port > PI_MAX_SOCKET_PORT))
|
|
SOFT_ERROR(PI_BAD_SOCKET_PORT, "bad port (%d)", port);
|
|
|
|
gpioCfg.socketPort = port;
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioCfgMemAlloc(unsigned memAllocMode)
|
|
{
|
|
DBG(DBG_USER, "memAllocMode=%d", memAllocMode);
|
|
|
|
CHECK_NOT_INITED;
|
|
|
|
if (memAllocMode > PI_MEM_ALLOC_MAILBOX)
|
|
SOFT_ERROR(
|
|
PI_BAD_MALLOC_MODE, "bad mem alloc mode (%d)", memAllocMode);
|
|
|
|
gpioCfg.memAllocMode = memAllocMode;
|
|
|
|
return 0;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
int gpioCfgNetAddr(int numSockAddr, uint32_t *sockAddr)
|
|
{
|
|
int i;
|
|
|
|
DBG(DBG_USER, "numSockAddr=%d sockAddr=%08"PRIXPTR,
|
|
numSockAddr, (uintptr_t)sockAddr);
|
|
|
|
CHECK_NOT_INITED;
|
|
|
|
if (numSockAddr <= 0) numSockNetAddr = 0;
|
|
else
|
|
{
|
|
if (numSockAddr >= MAX_CONNECT_ADDRESSES)
|
|
numSockAddr = MAX_CONNECT_ADDRESSES;
|
|
|
|
for (i=0; i<numSockAddr; i++) sockNetAddr[i] = sockAddr[i];
|
|
|
|
numSockNetAddr = numSockAddr;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
|
|
/* ----------------------------------------------------------------------- */
|
|
|
|
uint32_t gpioCfgGetInternals(void)
|
|
{
|
|
return gpioCfg.internals;
|
|
}
|
|
|
|
int gpioCfgSetInternals(uint32_t cfgVal)
|
|
{
|
|
if (libInitialised) return PI_INITIALISED;
|
|
gpioCfg.internals = cfgVal;
|
|
gpioCfg.dbgLevel = cfgVal & 0xF;
|
|
gpioCfg.alertFreq = (cfgVal>>4) & 0xF;
|
|
return 0;
|
|
}
|
|
|
|
|
|
/* include any user customisations */
|
|
|
|
#include "custom.cext"
|
|
|