core: add support for SET_CONFIG
authorKent Gibson <warthog618@gmail.com>
Sun, 1 Dec 2019 03:23:54 +0000 (11:23 +0800)
committerBartosz Golaszewski <bgolaszewski@baylibre.com>
Tue, 10 Dec 2019 10:45:52 +0000 (11:45 +0100)
Extend the libgpiod API to support the setting line configuration using the
GPIO GPIOHANDLE_SET_CONFIG_IOCTL uAPI ioctl.

The core change is the addition of gpiod_line_set_config, which provides a
low level wrapper around the ioctl.

Additionally, higher level helper functions, gpiod_line_set_flags,
gpiod_line_set_direction_input, and gpiod_line_set_direction_output provide
slightly simplified APIs for common use cases.

Bulk forms of all functions are also provided.

Documented the fields of gpiod_line to better identify the purpose of
each where the field name alone is not sufficiently clear, and to
indicate which flags are applicable to each field.

Implementation includes a few helper functions that serve to keep the
code tidier and are consistent with similar helper functions already
present.

Signed-off-by: Kent Gibson <warthog618@gmail.com>
Signed-off-by: Bartosz Golaszewski <bgolaszewski@baylibre.com>
include/gpiod.h
lib/core.c

index 984760460eee113c9555934cda66addbab99cd56..03fd4daf4682b70fb9d342a482cabebb220c1970 100644 (file)
@@ -1305,6 +1305,119 @@ int gpiod_line_set_value(struct gpiod_line *line, int value) GPIOD_API;
 int gpiod_line_set_value_bulk(struct gpiod_line_bulk *bulk,
                              const int *values) GPIOD_API;
 
+/**
+ * @}
+ *
+ * @defgroup __line_config__ Setting line configuration
+ * @{
+ */
+
+/**
+ * @brief Update the configuration of a single GPIO line.
+ * @param line GPIO line object.
+ * @param direction Updated direction which may be one of
+ *                  GPIOD_LINE_REQUEST_DIRECTION_AS_IS,
+ *                  GPIOD_LINE_REQUEST_DIRECTION_INPUT, or
+ *                  GPIOD_LINE_REQUEST_DIRECTION_OUTPUT.
+ * @param flags Replacement flags.
+ * @param value The new output value for the line when direction is
+ *              GPIOD_LINE_REQUEST_DIRECTION_OUTPUT.
+ * @return 0 is the operation succeeds. In case of an error this routine
+ *         returns -1 and sets the last error number.
+ */
+int gpiod_line_set_config(struct gpiod_line *line, int direction,
+                         int flags, int value) GPIOD_API;
+
+/**
+ * @brief Update the configuration of a set of GPIO lines.
+ * @param bulk Set of GPIO lines.
+ * @param direction Updated direction which may be one of
+ *                  GPIOD_LINE_REQUEST_DIRECTION_AS_IS,
+ *                  GPIOD_LINE_REQUEST_DIRECTION_INPUT, or
+ *                  GPIOD_LINE_REQUEST_DIRECTION_OUTPUT.
+ * @param flags Replacement flags.
+ * @param values An array holding line_bulk->num_lines new logical values
+ *               for lines when direction is
+ *               GPIOD_LINE_REQUEST_DIRECTION_OUTPUT.
+ *               A NULL pointer is interpreted as a logical low for all lines.
+ * @return 0 is the operation succeeds. In case of an error this routine
+ *         returns -1 and sets the last error number.
+ *
+ * If the lines were not previously requested together, the behavior is
+ * undefined.
+ */
+int gpiod_line_set_config_bulk(struct gpiod_line_bulk *bulk,
+                              int direction, int flags,
+                              const int *values) GPIOD_API;
+
+
+/**
+ * @brief Update the configuration flags of a single GPIO line.
+ * @param line GPIO line object.
+ * @param flags Replacement flags.
+ * @return 0 is the operation succeeds. In case of an error this routine
+ *         returns -1 and sets the last error number.
+ */
+int gpiod_line_set_flags(struct gpiod_line *line, int flags) GPIOD_API;
+
+/**
+ * @brief Update the configuration flags of a set of GPIO lines.
+ * @param bulk Set of GPIO lines.
+ * @param flags Replacement flags.
+ * @return 0 is the operation succeeds. In case of an error this routine
+ *         returns -1 and sets the last error number.
+ *
+ * If the lines were not previously requested together, the behavior is
+ * undefined.
+ */
+int gpiod_line_set_flags_bulk(struct gpiod_line_bulk *bulk,
+                             int flags) GPIOD_API;
+
+/**
+ * @brief Set the direction of a single GPIO line to input.
+ * @param line GPIO line object.
+ * @return 0 is the operation succeeds. In case of an error this routine
+ *         returns -1 and sets the last error number.
+ */
+int gpiod_line_set_direction_input(struct gpiod_line *line) GPIOD_API;
+
+/**
+ * @brief Set the direction of a set of GPIO lines to input.
+ * @param bulk Set of GPIO lines.
+ * @return 0 is the operation succeeds. In case of an error this routine
+ *         returns -1 and sets the last error number.
+ *
+ * If the lines were not previously requested together, the behavior is
+ * undefined.
+ */
+int
+gpiod_line_set_direction_input_bulk(struct gpiod_line_bulk *bulk) GPIOD_API;
+
+/**
+ * @brief Set the direction of a single GPIO line to output.
+ * @param line GPIO line object.
+ * @param value The logical value output on the line.
+ * @return 0 is the operation succeeds. In case of an error this routine
+ *         returns -1 and sets the last error number.
+ */
+int gpiod_line_set_direction_output(struct gpiod_line *line,
+                                   int value) GPIOD_API;
+
+/**
+ * @brief Set the direction of a set of GPIO lines to output.
+ * @param bulk Set of GPIO lines.
+ * @param values An array holding line_bulk->num_lines new logical values
+ *               for lines.  A NULL pointer is interpreted as a logical low
+ *               for all lines.
+ * @return 0 is the operation succeeds. In case of an error this routine
+ *         returns -1 and sets the last error number.
+ *
+ * If the lines were not previously requested together, the behavior is
+ * undefined.
+ */
+int gpiod_line_set_direction_output_bulk(struct gpiod_line_bulk *bulk,
+                                        const int *values) GPIOD_API;
+
 /**
  * @}
  *
index 0465de9bb639f57e85e3dcdda2fcbf4316fac2f0..b9fd6f42a88a4dd742ac403d90cc75479eebebb7 100644 (file)
@@ -34,10 +34,26 @@ struct line_fd_handle {
 
 struct gpiod_line {
        unsigned int offset;
+
+       /* The direction of the GPIO line. */
        int direction;
+
+       /* The active-state configuration. */
        int active_state;
+
+       /* The logical value last written to the line. */
+       int output_value;
+
+       /* The GPIOLINE_FLAGs returned by GPIO_GET_LINEINFO_IOCTL. */
        __u32 info_flags;
 
+       /* The GPIOD_LINE_REQUEST_FLAGs provided to request the line. */
+       __u32 req_flags;
+
+       /*
+        * Indicator of LINE_FREE, LINE_REQUESTED_VALUES or
+        * LINE_REQUESTED_EVENTS.
+        */
        int state;
 
        struct gpiod_chip *chip;
@@ -445,6 +461,20 @@ static bool line_bulk_all_requested(struct gpiod_line_bulk *bulk)
        return true;
 }
 
+static bool line_bulk_all_requested_values(struct gpiod_line_bulk *bulk)
+{
+       struct gpiod_line *line, **lineptr;
+
+       gpiod_line_bulk_foreach_line(bulk, line, lineptr) {
+               if (line->state != LINE_REQUESTED_VALUES) {
+                       errno = EPERM;
+                       return false;
+               }
+       }
+
+       return true;
+}
+
 static bool line_bulk_all_free(struct gpiod_line_bulk *bulk)
 {
        struct gpiod_line *line, **lineptr;
@@ -459,6 +489,27 @@ static bool line_bulk_all_free(struct gpiod_line_bulk *bulk)
        return true;
 }
 
+static bool line_request_direction_is_valid(int direction)
+{
+       if ((direction == GPIOD_LINE_REQUEST_DIRECTION_AS_IS) ||
+           (direction == GPIOD_LINE_REQUEST_DIRECTION_INPUT) ||
+           (direction == GPIOD_LINE_REQUEST_DIRECTION_OUTPUT))
+               return true;
+
+       errno = EINVAL;
+       return false;
+}
+
+static __u32 line_request_direction_to_gpio_handleflag(int direction)
+{
+       if (direction == GPIOD_LINE_REQUEST_DIRECTION_INPUT)
+               return GPIOHANDLE_REQUEST_INPUT;
+       if (direction == GPIOD_LINE_REQUEST_DIRECTION_OUTPUT)
+               return GPIOHANDLE_REQUEST_OUTPUT;
+
+       return 0;
+}
+
 static __u32 line_request_flag_to_gpio_handleflag(int flags)
 {
        int hflags = 0;
@@ -483,7 +534,7 @@ static int line_request_values(struct gpiod_line_bulk *bulk,
                               const struct gpiod_line_request_config *config,
                               const int *default_vals)
 {
-       struct gpiod_line *line, **lineptr;
+       struct gpiod_line *line;
        struct line_fd_handle *line_fd;
        struct gpiohandle_request req;
        unsigned int i;
@@ -536,8 +587,12 @@ static int line_request_values(struct gpiod_line_bulk *bulk,
        if (!line_fd)
                return -1;
 
-       gpiod_line_bulk_foreach_line(bulk, line, lineptr) {
+       gpiod_line_bulk_foreach_line_off(bulk, line, i) {
                line->state = LINE_REQUESTED_VALUES;
+               line->req_flags = config->flags;
+               if (config->request_type ==
+                               GPIOD_LINE_REQUEST_DIRECTION_OUTPUT)
+                       line->output_value = req.default_values[i];
                line_set_fd(line, line_fd);
 
                rv = gpiod_line_update(line);
@@ -583,6 +638,7 @@ static int line_request_event_single(struct gpiod_line *line,
                return -1;
 
        line->state = LINE_REQUESTED_EVENTS;
+       line->req_flags = config->flags;
        line_set_fd(line, line_fd);
 
        rv = gpiod_line_update(line);
@@ -766,9 +822,131 @@ int gpiod_line_set_value_bulk(struct gpiod_line_bulk *bulk, const int *values)
        if (rv < 0)
                return -1;
 
+       gpiod_line_bulk_foreach_line_off(bulk, line, i)
+               line->output_value = data.values[i];
+
+       return 0;
+}
+
+int gpiod_line_set_config(struct gpiod_line *line, int direction,
+                         int flags, int value)
+{
+       struct gpiod_line_bulk bulk;
+
+       gpiod_line_bulk_init(&bulk);
+       gpiod_line_bulk_add(&bulk, line);
+
+       return gpiod_line_set_config_bulk(&bulk, direction, flags, &value);
+}
+
+int gpiod_line_set_config_bulk(struct gpiod_line_bulk *bulk,
+                              int direction, int flags,
+                              const int *values)
+{
+       struct gpiohandle_config hcfg;
+       struct gpiod_line *line;
+       unsigned int i;
+       int rv, fd;
+
+       if (!line_bulk_same_chip(bulk) ||
+           !line_bulk_all_requested_values(bulk))
+               return -1;
+
+       if (!line_request_direction_is_valid(direction))
+               return -1;
+
+       memset(&hcfg, 0, sizeof(hcfg));
+
+       hcfg.flags = line_request_flag_to_gpio_handleflag(flags);
+       hcfg.flags |= line_request_direction_to_gpio_handleflag(direction);
+       if (direction == GPIOD_LINE_REQUEST_DIRECTION_OUTPUT && values) {
+               for (i = 0; i < gpiod_line_bulk_num_lines(bulk); i++)
+                       hcfg.default_values[i] = (uint8_t)!!values[i];
+       }
+
+       line = gpiod_line_bulk_get_line(bulk, 0);
+       fd = line_get_fd(line);
+
+       rv = ioctl(fd, GPIOHANDLE_SET_CONFIG_IOCTL, &hcfg);
+       if (rv < 0)
+               return -1;
+
+       gpiod_line_bulk_foreach_line_off(bulk, line, i) {
+               line->req_flags = flags;
+               if (direction == GPIOD_LINE_REQUEST_DIRECTION_OUTPUT)
+                       line->output_value = hcfg.default_values[i];
+
+               rv = gpiod_line_update(line);
+               if (rv < 0)
+                       return rv;
+       }
        return 0;
 }
 
+int gpiod_line_set_flags(struct gpiod_line *line, int flags)
+{
+       struct gpiod_line_bulk bulk;
+
+       gpiod_line_bulk_init(&bulk);
+       gpiod_line_bulk_add(&bulk, line);
+
+       return gpiod_line_set_flags_bulk(&bulk, flags);
+}
+
+int gpiod_line_set_flags_bulk(struct gpiod_line_bulk *bulk, int flags)
+{
+       struct gpiod_line *line;
+       int values[GPIOD_LINE_BULK_MAX_LINES];
+       unsigned int i;
+       int direction;
+
+       line = gpiod_line_bulk_get_line(bulk, 0);
+       if (line->direction == GPIOD_LINE_DIRECTION_OUTPUT) {
+               gpiod_line_bulk_foreach_line_off(bulk, line, i)
+                       values[i] = line->output_value;
+
+               direction = GPIOD_LINE_REQUEST_DIRECTION_OUTPUT;
+       } else {
+               direction = GPIOD_LINE_REQUEST_DIRECTION_INPUT;
+       }
+
+       return gpiod_line_set_config_bulk(bulk, direction,
+                                         flags, values);
+}
+
+int gpiod_line_set_direction_input(struct gpiod_line *line)
+{
+       return gpiod_line_set_config(line, GPIOD_LINE_REQUEST_DIRECTION_INPUT,
+                                    line->req_flags, 0);
+}
+
+int gpiod_line_set_direction_input_bulk(struct gpiod_line_bulk *bulk)
+{
+       struct gpiod_line *line;
+
+       line = gpiod_line_bulk_get_line(bulk, 0);
+       return gpiod_line_set_config_bulk(bulk,
+                                         GPIOD_LINE_REQUEST_DIRECTION_INPUT,
+                                         line->req_flags, NULL);
+}
+
+int gpiod_line_set_direction_output(struct gpiod_line *line, int value)
+{
+       return gpiod_line_set_config(line, GPIOD_LINE_REQUEST_DIRECTION_OUTPUT,
+                                    line->req_flags, value);
+}
+
+int gpiod_line_set_direction_output_bulk(struct gpiod_line_bulk *bulk,
+                                        const int *values)
+{
+       struct gpiod_line *line;
+
+       line = gpiod_line_bulk_get_line(bulk, 0);
+       return gpiod_line_set_config_bulk(bulk,
+                                         GPIOD_LINE_REQUEST_DIRECTION_OUTPUT,
+                                         line->req_flags, values);
+}
+
 int gpiod_line_event_wait(struct gpiod_line *line,
                          const struct timespec *timeout)
 {