#include "rtapi.h"
#include "rtapi_app.h"
#include "hal.h"
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <sys/mman.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include "beaglebone_gpio.h"
#define MODNAME "hal_bb_gpio"
MODULE_AUTHOR("Ian McMahon");
MODULE_DESCRIPTION("Driver for BeagleBone GPIO pins");
MODULE_LICENSE("GPL");
#define HEADERS 2
#define PINS_PER_HEADER 46
typedef struct {
hal_bit_t* led_pins[4];
hal_bit_t* input_pins[1 + PINS_PER_HEADER * HEADERS]; hal_bit_t* output_pins[1 + PINS_PER_HEADER * HEADERS]; hal_bit_t *led_inv[4];
hal_bit_t *input_inv[1 + PINS_PER_HEADER * HEADERS];
hal_bit_t *output_inv[1 + PINS_PER_HEADER * HEADERS];
} port_data_t;
static port_data_t *port_data;
static const char *modname = MODNAME;
static void write_port(void *arg, long period);
static void read_port(void *arg, long period);
static off_t start_addr_for_port(int port);
static void configure_pin(bb_gpio_pin *pin, char mode);
static int comp_id;
static int num_ports;
static char *user_leds;
RTAPI_MP_STRING(user_leds, "user leds, comma separated. 0-3");
static char *input_pins;
RTAPI_MP_STRING(input_pins, "input pins, comma separated. P8 pins add 800, P9 pins add 900");
static char *output_pins;
RTAPI_MP_STRING(output_pins, "output pins, comma separated. P8 pins add 800, P9 pins add 900");
int configure_control_module() {
int fd = rtapi_open_as_root("/dev/mem", O_RDWR);
control_module = mmap(0, CONTROL_MODULE_SIZE, PROT_READ | PROT_WRITE, MAP_SHARED, fd, CONTROL_MODULE_START_ADDR);
if(control_module == MAP_FAILED) {
rtapi_print_msg(RTAPI_MSG_ERR, "%s: ERROR: Unable to map Control Module: %s\n", modname, strerror(errno));
return -errno;
}
close(fd);
return 0;
}
int configure_gpio_port(int n) {
volatile void *cm_per; volatile unsigned int *regptr;
unsigned int regvalue;
int fd = rtapi_open_as_root("/dev/mem", O_RDWR);
gpio_ports[n] = hal_malloc(sizeof(bb_gpio_port));
if ( n > 0 ) {
cm_per = mmap(0,CM_PER_LEN, PROT_READ | PROT_WRITE, MAP_SHARED, fd, CM_PER_ADDR);
if(cm_per == MAP_FAILED) {
rtapi_print_msg(RTAPI_MSG_ERR, "%s: ERROR: Unable to map Clock Module: %s\n", modname, strerror(errno));
return -errno;
}
regptr = cm_per + CM_PER_GPIO1_CLKCTRL_OFFSET + 4*(n-1);
regvalue = *regptr;
if ( (regvalue & CM_PER_GPIO_CLKCTRL_MODMODE_MASK ) != CM_PER_GPIO_CLKCTRL_MODMODE_ENABLED ) {
rtapi_print_msg(RTAPI_MSG_ERR, "%s: ERROR: GPIO Port %d is not enabled in device tree\n", modname, n);
return -ENODEV;
}
munmap((void *)cm_per, CM_PER_LEN);
}
gpio_ports[n]->gpio_addr = mmap(0, GPIO_SIZE, PROT_READ | PROT_WRITE, MAP_SHARED, fd, start_addr_for_port(n));
if(gpio_ports[n]->gpio_addr == MAP_FAILED) {
rtapi_print_msg(RTAPI_MSG_ERR, "%s: ERROR: Unable to map GPIO: %s\n", modname, strerror(errno));
return -errno;
}
gpio_ports[n]->oe_reg = gpio_ports[n]->gpio_addr + GPIO_OE;
gpio_ports[n]->setdataout_reg = gpio_ports[n]->gpio_addr + GPIO_SETDATAOUT;
gpio_ports[n]->clrdataout_reg = gpio_ports[n]->gpio_addr + GPIO_CLEARDATAOUT;
gpio_ports[n]->datain_reg = gpio_ports[n]->gpio_addr + GPIO_DATAIN;
rtapi_print("memmapped gpio port %d to %p, oe: %p, set: %p, clr: %p\n", n, gpio_ports[n]->gpio_addr, gpio_ports[n]->oe_reg, gpio_ports[n]->setdataout_reg, gpio_ports[n]->clrdataout_reg);
close(fd);
return 0;
}
int rtapi_app_main(void) {
char name[HAL_NAME_LEN + 1];
int n, retval;
char *data, *token;
num_ports = 1;
n = 0;
comp_id = hal_init(modname);
if(comp_id < 0) {
rtapi_print_msg(RTAPI_MSG_ERR, "%s: ERROR: hal_init() failed\n", modname);
return -1;
}
port_data = hal_malloc(num_ports * sizeof(port_data_t));
if(port_data == 0) {
rtapi_print_msg(RTAPI_MSG_ERR, "%s: ERROR: hal_malloc() failed\n", modname);
hal_exit(comp_id);
return -1;
}
int result = configure_control_module();
if(result < 0) {
hal_exit(comp_id);
return result;
}
if(user_leds != NULL) {
data = user_leds;
while((token = strtok(data, ",")) != NULL) {
int led = strtol(token, NULL, 10);
data = NULL;
if(user_led_gpio_pins[led].claimed != 0) {
rtapi_print_msg(RTAPI_MSG_ERR, "%s: ERROR: userled%d is not available as a GPIO.\n", modname, led);
hal_exit(comp_id);
return -1;
}
retval = hal_pin_bit_newf(HAL_IN, &(port_data->led_pins[led]), comp_id, "bb_gpio.userled%d", led);
if(retval < 0) {
rtapi_print_msg(RTAPI_MSG_ERR, "%s: ERROR: userled %d could not export pin, err: %d\n", modname, led, retval);
hal_exit(comp_id);
return -1;
}
retval = hal_pin_bit_newf(HAL_IN, &(port_data->led_inv[led]), comp_id, "bb_gpio.userled%d-invert", led);
if(retval < 0) {
rtapi_print_msg(RTAPI_MSG_ERR, "%s: ERROR: userled %d could not export pin, err: %d\n", modname, led, retval);
hal_exit(comp_id);
return -1;
}
*(port_data->led_inv[led]) = 0;
int gpio_num = user_led_gpio_pins[led].port_num;
if(gpio_ports[gpio_num] == NULL) {
int result = configure_gpio_port(gpio_num);
if(result < 0) {
hal_exit(comp_id);
return result;
}
}
user_led_gpio_pins[led].port = gpio_ports[gpio_num];
configure_pin(&user_led_gpio_pins[led], 'O');
}
}
if(input_pins != NULL) {
data = input_pins;
while((token = strtok(data, ",")) != NULL) {
int pin = strtol(token, NULL, 10);
int header;
bb_gpio_pin *bbpin;
if (pin < 300)
pin += 700;
if(pin < 801 || pin > 946 || (pin > 846 && pin < 901)) {
rtapi_print_msg(RTAPI_MSG_ERR, "%s: ERROR: invalid pin number '%d'. Valid pins are 801-846 for P8 pins, 901-946 for P9 pins.\n", modname, pin);
hal_exit(comp_id);
return -1;
}
if(pin < 900) {
pin -= 800;
bbpin = &p8_pins[pin];
header = 8;
} else {
pin -= 900;
bbpin = &p9_pins[pin];
header = 9;
}
if(bbpin->claimed != 0) {
rtapi_print_msg(RTAPI_MSG_ERR, "%s: ERROR: pin p%d.%02d is not available as a GPIO.\n", modname, header, pin);
hal_exit(comp_id);
return -1;
}
data = NULL;
retval = hal_pin_bit_newf(HAL_OUT, &(port_data->input_pins[pin + (header - 8)*PINS_PER_HEADER]), comp_id, "bb_gpio.p%d.in-%02d", header, pin);
if(retval < 0) {
rtapi_print_msg(RTAPI_MSG_ERR, "%s: ERROR: pin p%d.%02d could not export pin, err: %d\n", modname, header, pin, retval);
hal_exit(comp_id);
return -1;
}
retval = hal_pin_bit_newf(HAL_IN, &(port_data->input_inv[pin + (header - 8)*PINS_PER_HEADER]), comp_id, "bb_gpio.p%d.in-%02d-invert", header, pin);
if(retval < 0) {
rtapi_print_msg(RTAPI_MSG_ERR, "%s: ERROR: pin p%d.%02d could not export pin, err: %d\n", modname, header, pin, retval);
hal_exit(comp_id);
return -1;
}
*(port_data->input_inv[pin + (header - 8)*PINS_PER_HEADER]) = 0;
int gpio_num = bbpin->port_num;
if(gpio_ports[gpio_num] == NULL) {
int result = configure_gpio_port(gpio_num);
if(result < 0) {
hal_exit(comp_id);
return result;
}
}
bbpin->port = gpio_ports[gpio_num];
configure_pin(bbpin, 'U');
rtapi_print("pin %d maps to pin %d-%d, mode %d\n", pin, bbpin->port_num, bbpin->pin_num, bbpin->claimed);
}
}
if(output_pins != NULL) {
data = output_pins;
while((token = strtok(data, ",")) != NULL) {
int pin = strtol(token, NULL, 10);
int header;
bb_gpio_pin *bbpin;
if (pin < 300)
pin += 700;
if(pin < 801 || pin > 946 || (pin > 846 && pin < 901)) {
rtapi_print_msg(RTAPI_MSG_ERR, "%s: ERROR: invalid pin number '%d'. Valid pins are 801-846 for P8 pins, 901-946 for P9 pins.\n", modname, pin);
hal_exit(comp_id);
return -1;
}
if(pin < 900) {
pin -= 800;
bbpin = &p8_pins[pin];
header = 8;
} else {
pin -= 900;
bbpin = &p9_pins[pin];
header = 9;
}
if(bbpin->claimed != 0) {
rtapi_print_msg(RTAPI_MSG_ERR, "%s: ERROR: pin p%d.%02d is not available as a GPIO.\n", modname, header, pin);
hal_exit(comp_id);
return -1;
}
data = NULL;
retval = hal_pin_bit_newf(HAL_IN, &(port_data->output_pins[pin + (header - 8)*PINS_PER_HEADER]), comp_id, "bb_gpio.p%d.out-%02d", header, pin);
if(retval < 0) {
rtapi_print_msg(RTAPI_MSG_ERR, "%s: ERROR: pin p%d.%02d could not export pin, err: %d\n", modname, header, pin, retval);
hal_exit(comp_id);
return -1;
}
retval = hal_pin_bit_newf(HAL_IN, &(port_data->output_inv[pin + (header - 8)*PINS_PER_HEADER]), comp_id, "bb_gpio.p%d.out-%02d-invert", header, pin);
if(retval < 0) {
rtapi_print_msg(RTAPI_MSG_ERR, "%s: ERROR: pin p%d.%02d could not export pin, err: %d\n", modname, header, pin, retval);
hal_exit(comp_id);
return -1;
}
*(port_data->output_inv[pin + (header - 8)*PINS_PER_HEADER]) = 0;
int gpio_num = bbpin->port_num;
if(gpio_ports[gpio_num] == NULL) {
int result = configure_gpio_port(gpio_num);
if(result < 0) {
hal_exit(comp_id);
return result;
}
}
bbpin->port = gpio_ports[gpio_num];
configure_pin(bbpin, 'O');
}
}
rtapi_snprintf(name, sizeof(name), "bb_gpio.write");
retval = hal_export_funct(name, write_port, port_data, 0, 0, comp_id);
if(retval < 0) {
rtapi_print_msg(RTAPI_MSG_ERR, "%s: ERROR: port %d write funct export failed\n", modname, n);
hal_exit(comp_id);
return -1;
}
rtapi_snprintf(name, sizeof(name), "bb_gpio.read");
retval = hal_export_funct(name, read_port, port_data, 0, 0, comp_id);
if(retval < 0) {
rtapi_print_msg(RTAPI_MSG_ERR, "%s: ERROR: port %d read funct export failed\n", modname, n);
hal_exit(comp_id);
return -1;
}
rtapi_print_msg(RTAPI_MSG_INFO, "%s: installed driver\n", modname);
hal_ready(comp_id);
return 0;
}
void rtapi_app_exit(void) {
hal_exit(comp_id);
}
static void write_port(void *arg, long period) {
int i;
port_data_t *port = (port_data_t *)arg;
for(i=0; i<4; i++) {
if(port->led_pins[i] == NULL) continue;
bb_gpio_pin pin = user_led_gpio_pins[i];
if(pin.claimed != 'O') continue;
if((*port->led_pins[i] ^ *(port->led_inv[i])) == 0)
*(pin.port->clrdataout_reg) = (1 << pin.pin_num);
else
*(pin.port->setdataout_reg) = (1 << pin.pin_num);
}
for(i=1; i<=HEADERS*PINS_PER_HEADER; i++) {
if(port->output_pins[i] == NULL) continue;
bb_gpio_pin pin;
if(i<PINS_PER_HEADER)
pin = p8_pins[i];
else
pin = p9_pins[i - PINS_PER_HEADER];
if(pin.claimed != 'O') continue;
if((*port->output_pins[i] ^ *(port->output_inv[i])) == 0)
*(pin.port->clrdataout_reg) = (1 << pin.pin_num);
else
*(pin.port->setdataout_reg) = (1 << pin.pin_num);
}
}
static void read_port(void *arg, long period) {
int i;
port_data_t *port = (port_data_t *)arg;
for(i=1; i<=HEADERS*PINS_PER_HEADER; i++) {
if(port->input_pins[i] == NULL) continue;
bb_gpio_pin pin;
if(i<PINS_PER_HEADER) {
pin = p8_pins[i];
} else {
pin = p9_pins[i - PINS_PER_HEADER];
}
if(!(pin.claimed == 'I' || pin.claimed == 'U' || pin.claimed == 'D')) continue;
*port->input_pins[i] = ((*(pin.port->datain_reg) & (1 << pin.pin_num)) >> pin.pin_num) ^ *(port->input_inv[i]);
}
}
off_t start_addr_for_port(int port) {
switch(port) {
case 0:
return GPIO0_START_ADDR;
break;
case 1:
return GPIO1_START_ADDR;
break;
case 2:
return GPIO2_START_ADDR;
break;
case 3:
return GPIO3_START_ADDR;
break;
default:
return -1;
break;
}
}
void configure_pin(bb_gpio_pin *pin, char mode) {
volatile unsigned int *control_reg = control_module + pin->control_offset;
pin->claimed = mode;
switch(mode) {
case 'O':
*(pin->port->oe_reg) &= ~(1 << pin->pin_num); *control_reg = PIN_MODE7 | PIN_PULLUD_DISABLED | PIN_RX_DISABLED;
break;
case 'I':
*(pin->port->oe_reg) |= (1 << pin->pin_num); *control_reg = PIN_MODE7 | PIN_PULLUD_DISABLED | PIN_RX_ENABLED;
break;
case 'U':
*(pin->port->oe_reg) |= (1 << pin->pin_num); *control_reg = PIN_MODE7 | PIN_PULLUD_ENABLED | PIN_PULLUP | PIN_RX_ENABLED;
break;
case 'D':
*(pin->port->oe_reg) |= (1 << pin->pin_num); *control_reg = PIN_MODE7 | PIN_PULLUD_ENABLED | PIN_PULLDOWN | PIN_RX_ENABLED;
break;
default:
break;
}
}