/* * libftdi1 sandbox barebox GPIO driver * * Copyright (C) 2016, 2017 Antony Pavlov * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License as * published by the Free Software Foundation; either version 2 of * the License, or (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * */ #include #include #include #include #include #include struct libftdi1_gpio_chip { struct gpio_chip chip; struct ft2232_bitbang *ftbb; }; static int libftdi1_gpio_direction_input(struct gpio_chip *chip, unsigned off) { struct libftdi1_gpio_chip *gpio = container_of(chip, struct libftdi1_gpio_chip, chip); barebox_libftdi1_gpio_direction(gpio->ftbb, off, GPIOF_DIR_IN); barebox_libftdi1_update(gpio->ftbb); return 0; } static int libftdi1_gpio_direction_output( struct gpio_chip *chip, unsigned off, int value) { struct libftdi1_gpio_chip *gpio = container_of(chip, struct libftdi1_gpio_chip, chip); barebox_libftdi1_gpio_set_value(gpio->ftbb, off, value); barebox_libftdi1_gpio_direction(gpio->ftbb, off, GPIOF_DIR_OUT); barebox_libftdi1_update(gpio->ftbb); return 0; } static int libftdi1_gpio_get_value(struct gpio_chip *chip, unsigned off) { struct libftdi1_gpio_chip *gpio = container_of(chip, struct libftdi1_gpio_chip, chip); return barebox_libftdi1_gpio_get_value(gpio->ftbb, off); } static void libftdi1_gpio_set_value( struct gpio_chip *chip, unsigned off, int value) { struct libftdi1_gpio_chip *gpio = container_of(chip, struct libftdi1_gpio_chip, chip); barebox_libftdi1_gpio_set_value(gpio->ftbb, off, value); barebox_libftdi1_update(gpio->ftbb); } static struct gpio_ops libftdi1_gpio_ops = { .direction_input = libftdi1_gpio_direction_input, .direction_output = libftdi1_gpio_direction_output, .get = libftdi1_gpio_get_value, .set = libftdi1_gpio_set_value, }; static int libftdi1_gpio_probe(struct device_d *dev) { struct libftdi1_gpio_chip *gpio; struct ft2232_bitbang *ftbb; int ret; uint32_t id_vendor, id_product; const char *i_serial_number = NULL; of_property_read_u32(dev->device_node, "usb,id_vendor", &id_vendor); of_property_read_u32(dev->device_node, "usb,id_product", &id_product); of_property_read_string(dev->device_node, "usb,i_serial_number", &i_serial_number); ftbb = barebox_libftdi1_open(id_vendor, id_product, i_serial_number); if (!ftbb) return -EIO; gpio = xzalloc(sizeof(*gpio)); gpio->ftbb = ftbb; gpio->chip.dev = dev; gpio->chip.ops = &libftdi1_gpio_ops; gpio->chip.base = 0; gpio->chip.ngpio = 8; ret = gpiochip_add(&gpio->chip); dev_dbg(dev, "%d: probed gpio%d with base %d\n", ret, dev->id, gpio->chip.base); return 0; } static __maybe_unused const struct of_device_id libftdi1_gpio_dt_ids[] = { { .compatible = "barebox,libftdi1-gpio", }, { /* sentinel */ }, }; static void libftdi1_gpio_remove(struct device_d *dev) { barebox_libftdi1_close(); } static struct driver_d libftdi1_gpio_driver = { .name = "libftdi1-gpio", .probe = libftdi1_gpio_probe, .remove = libftdi1_gpio_remove, .of_compatible = DRV_OF_COMPAT(libftdi1_gpio_dt_ids), }; device_platform_driver(libftdi1_gpio_driver);