firmware/general/package/sigmastar-motors/src/motor.c

166 lines
3.2 KiB
C

#include <errno.h>
#include <fcntl.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <linux/gpio.h>
#include <sys/file.h>
#include <sys/ioctl.h>
#define DEV_NAME "/dev/gpiochip0"
#define STEP_TIME 1000
#define STEP_COUNT 20
#define SEQ_COUNT 8
typedef struct {
const char *name;
int gpio_x[4];
int gpio_y[4];
} config;
static config list[] = {
{ "rtl8188fu-ssc337de-foscam", { 01, 02, 12, 13 }, { 62, 63, 64, 65 }, },
};
static int sequence[][4] = {
{ 1, 0, 0, 0 }, { 1, 1, 0, 0 }, { 0, 1, 0, 0 }, { 0, 1, 1, 0 },
{ 0, 0, 1, 0 }, { 0, 0, 1, 1 }, { 0, 0, 0, 1 }, { 1, 0, 0, 1 },
{ 1, 0, 0, 1 }, { 0, 0, 0, 1 }, { 0, 0, 1, 1 }, { 0, 0, 1, 0 },
{ 0, 1, 1, 0 }, { 0, 1, 0, 0 }, { 1, 1, 0, 0 }, { 1, 0, 0, 0 },
};
static int write_gpio(int pin, int val) {
struct gpiohandle_request rq;
struct gpiohandle_data data;
int fd = open(DEV_NAME, O_RDONLY);
if (fd < 0) {
printf("Unable to open chip: %s\n", strerror(errno));
return 1;
}
rq.lineoffsets[0] = pin;
rq.flags = GPIOHANDLE_REQUEST_OUTPUT;
rq.lines = 1;
if (ioctl(fd, GPIO_GET_LINEHANDLE_IOCTL, &rq) < 0) {
printf("Unable to request gpio %d: %s\n", pin, strerror(errno));
return 1;
}
close(fd);
data.values[0] = val;
if (ioctl(rq.fd, GPIOHANDLE_SET_LINE_VALUES_IOCTL, &data) < 0) {
printf("Unable to set value %d: %s\n", val, strerror(errno));
return 1;
}
close(rq.fd);
return 0;
}
static int motor_control(int *gpio, int count) {
for (int i = count; i < count + SEQ_COUNT; i++) {
for (int j = 0; j < 4; j++) {
if (write_gpio(gpio[j], sequence[i][j])) {
return 1;
}
usleep(STEP_TIME);
}
}
return 0;
}
static int gpio_export(int *gpio) {
for (int i = 0; i < 4; i++) {
if (write_gpio(gpio[i], 0)) {
return 1;
}
}
return 0;
}
static void print_list(int size) {
char buffer[1024];
int len = 0;
for (int i = 0; i < size; i++) {
len += snprintf(buffer + len, sizeof(buffer), "%s\n", list[i].name);
}
printf("%s", buffer);
}
int main(int argc, char **argv) {
int size = sizeof(list) / sizeof(config);
if (argc > 1 && strstr(argv[1], "list")) {
print_list(size);
return -1;
} else if (argc < 3 || argc > 4) {
printf("Usage: %s [device] [x_step] [y_step]\n", argv[0]);
return -1;
}
int dev = -1;
for (int i = 0; i < sizeof(list) / sizeof(config); i++) {
if (strstr(argv[1], list[i].name)) {
dev = i;
break;
}
}
if (dev < 0) {
printf("Device not supported\n");
return -1;
}
int pid = open("/var/run/motor.pid", O_RDWR | O_CREAT, 0644);
if (flock(pid, LOCK_EX | LOCK_NB)) {
printf("Control in progress\n");
close(pid);
return -1;
}
int x = argv[2] ? atoi(argv[2]) : 0;
int y = argv[3] ? atoi(argv[3]) : 0;
if (gpio_export(list[dev].gpio_x) || gpio_export(list[dev].gpio_y)) {
close(pid);
return -1;
}
int count_x = abs(x) ? STEP_COUNT : 0;
int count_y = abs(y) ? STEP_COUNT : 0;
while (count_x || count_y) {
if (count_x) {
if (motor_control(list[dev].gpio_x, (x < 0) ? SEQ_COUNT : 0)) {
goto reset;
}
count_x--;
}
if (count_y) {
if (motor_control(list[dev].gpio_y, (y < 0) ? 0 : SEQ_COUNT)) {
goto reset;
}
count_y--;
}
}
reset:
gpio_export(list[dev].gpio_x);
gpio_export(list[dev].gpio_y);
close(pid);
return 0;
}