From 2dcd065bb05ce40d31b6bfd2d3fb2b1d52e22f88 Mon Sep 17 00:00:00 2001
From: trilader <trilader@schroedingers-bit.net>
Date: Mon, 19 Aug 2019 10:38:19 +0200
Subject: [PATCH] feat(pycardium): Add (wristband) GPIO module

---
 Documentation/index.rst          |  1 +
 Documentation/pycardium/gpio.rst | 81 ++++++++++++++++++++++++++++
 pycardium/meson.build            |  1 +
 pycardium/modules/gpio.c         | 93 ++++++++++++++++++++++++++++++++
 pycardium/modules/qstrdefs.h     | 15 ++++++
 pycardium/mpconfigport.h         |  1 +
 6 files changed, 192 insertions(+)
 create mode 100644 Documentation/pycardium/gpio.rst
 create mode 100644 pycardium/modules/gpio.c

diff --git a/Documentation/index.rst b/Documentation/index.rst
index 27b0d56a..6ca720da 100644
--- a/Documentation/index.rst
+++ b/Documentation/index.rst
@@ -24,6 +24,7 @@ Last but not least, if you want to start hacking the lower-level firmware, the
    pycardium/buttons
    pycardium/color
    pycardium/display
+   pycardium/gpio
    pycardium/leds
    pycardium/light-sensor
    pycardium/os
diff --git a/Documentation/pycardium/gpio.rst b/Documentation/pycardium/gpio.rst
new file mode 100644
index 00000000..abcda8de
--- /dev/null
+++ b/Documentation/pycardium/gpio.rst
@@ -0,0 +1,81 @@
+.. py:module:: gpio
+
+``gpio`` - GPIO Pins
+==========================
+The :py:mod:`gpio` module allows you to use card10's GPIO pins as input and
+output in your scripts.
+
+**Example**:
+
+.. code-block:: python
+
+   import gpio
+
+   gpio.set_mode(gpio.WRISTBAND_1, gpio.mode.OUTPUT)
+   gpio.write(gpio.WRISTBAND_1, True)
+   
+   gpio.set_mode(gpio.WRISTBAND_2, gpio.mode.INPUT | gpio.mode.PULL_UP)
+   state = gpio.read(gpio.WRISTBAND_2)
+   print("State of Wristband pin 2:", state)
+
+.. py:function:: gpio.set_mode(pin, mode)
+
+   Configure GPIO pin state.
+
+   :param int pin: ID of the pin to be configured.
+   :param int mode: An integer with the bits for the wanted mode set. Create your
+      integer by ORing :py:data:`gpio.mode.OUTPUT`, :py:data:`gpio.mode.INPUT`,
+      :py:data:`gpio.mode.PULL_UP`, :py:data:`gpio.mode.PULL_DOWN`.
+
+.. py:function:: gpio.get_mode(pin)
+
+   Get GPIO pin state.
+   
+   :param int pin: ID of the pin of to get the mode of.
+   :returns: An integer with the configure mode bits set.
+
+.. py:function:: gpio.write(pin, value)
+
+   Write a value to a GPIO pin.
+   
+   :param int pin: ID of the pin of to get the mode of.
+   :param bool value: New pin value.
+   
+.. py:function:: gpio.read(pin)
+
+   Read GPIO pin value.
+   
+   :param int pin: ID of the pin of to get the mode of.
+   :returns: Current value of the GPIO pin.
+
+.. py:data:: gpio.WRISTBAND_1
+
+   Pin ID for Wristband GPIO 1.
+
+.. py:data:: gpio.WRISTBAND_2
+
+   Pin ID for Wristband GPIO 2.
+
+.. py:data:: gpio.WRISTBAND_3
+
+   Pin ID for Wristband GPIO 3.
+
+.. py:data:: gpio.WRISTBAND_4
+
+   Pin ID for Wristband GPIO 4.
+   
+.. py:data:: gpio.mode.OUTPUT
+
+   Configures a pin as output.
+   
+.. py:data:: gpio.mode.INPUT
+
+   Configures a pin as input.
+   
+.. py:data:: gpio.mode:PULL_UP
+
+   Enables the internal pull-up resistor of a pin.
+   
+.. py:data:: gpio.mode:PULL_DOwn
+
+   Enables the internal pull-down resistor of a pin.
diff --git a/pycardium/meson.build b/pycardium/meson.build
index 4d976e0a..8d3b447a 100644
--- a/pycardium/meson.build
+++ b/pycardium/meson.build
@@ -4,6 +4,7 @@ modsrc = files(
   'modules/buttons.c',
   'modules/fat_file.c',
   'modules/fat_reader_import.c',
+  'modules/gpio.c',
   'modules/interrupt.c',
   'modules/sys_leds.c',
   'modules/light_sensor.c',
diff --git a/pycardium/modules/gpio.c b/pycardium/modules/gpio.c
new file mode 100644
index 00000000..c92d1023
--- /dev/null
+++ b/pycardium/modules/gpio.c
@@ -0,0 +1,93 @@
+#include "py/obj.h"
+#include "py/runtime.h"
+
+#include "epicardium.h"
+
+static mp_obj_t mp_gpio_set_mode(mp_obj_t pin_obj, mp_obj_t mode_obj)
+{
+	int pin  = mp_obj_get_int(pin_obj);
+	int mode = mp_obj_get_int(mode_obj);
+	int rc   = epic_gpio_set_pin_mode(pin, mode);
+	if (rc < 0)
+		mp_raise_OSError(-rc);
+
+	return mp_const_none;
+};
+static MP_DEFINE_CONST_FUN_OBJ_2(gpio_set_mode, mp_gpio_set_mode);
+
+static mp_obj_t mp_gpio_get_mode(mp_obj_t pin_obj)
+{
+	int pin  = mp_obj_get_int(pin_obj);
+	int mode = epic_gpio_get_pin_mode(pin);
+	if (mode < 0)
+		mp_raise_OSError(-mode);
+
+	return MP_OBJ_NEW_SMALL_INT(mode);
+};
+static MP_DEFINE_CONST_FUN_OBJ_1(gpio_get_mode, mp_gpio_get_mode);
+
+static mp_obj_t mp_gpio_read(mp_obj_t pin_obj)
+{
+	int pin        = mp_obj_get_int(pin_obj);
+	uint32_t value = epic_gpio_read_pin(pin);
+	if (value == -EINVAL)
+		mp_raise_OSError(EINVAL);
+
+	return MP_OBJ_NEW_SMALL_INT(value);
+};
+static MP_DEFINE_CONST_FUN_OBJ_1(gpio_read, mp_gpio_read);
+
+static mp_obj_t mp_gpio_write(mp_obj_t pin_obj, mp_obj_t on_obj)
+{
+	int pin = mp_obj_get_int(pin_obj);
+	bool on = mp_obj_is_true(on_obj);
+	int rc  = epic_gpio_write_pin(pin, on);
+	if (rc < 0)
+		mp_raise_OSError(-rc);
+
+	return mp_const_none;
+};
+static MP_DEFINE_CONST_FUN_OBJ_2(gpio_write, mp_gpio_write);
+
+static const mp_rom_map_elem_t gpio_module_modes_table[] = {
+	{ MP_ROM_QSTR(MP_QSTR___name__), MP_ROM_QSTR(MP_QSTR_mode) },
+	{ MP_ROM_QSTR(MP_QSTR_INPUT), MP_OBJ_NEW_SMALL_INT(GPIO_MODE_IN) },
+	{ MP_ROM_QSTR(MP_QSTR_OUTPUT), MP_OBJ_NEW_SMALL_INT(GPIO_MODE_OUT) },
+	{ MP_ROM_QSTR(MP_QSTR_PULL_UP), MP_OBJ_NEW_SMALL_INT(GPIO_PULL_UP) },
+	{ MP_ROM_QSTR(MP_QSTR_PULL_DOWN),
+	  MP_OBJ_NEW_SMALL_INT(GPIO_PULL_DOWN) },
+};
+static MP_DEFINE_CONST_DICT(gpio_module_modes_dict, gpio_module_modes_table);
+
+const mp_obj_module_t gpio_module_modes = {
+	.base    = { &mp_type_module },
+	.globals = (mp_obj_dict_t *)&gpio_module_modes_dict,
+};
+
+/* The globals table for this module */
+static const mp_rom_map_elem_t gpio_module_globals_table[] = {
+	{ MP_ROM_QSTR(MP_QSTR___name__), MP_ROM_QSTR(MP_QSTR_gpio) },
+	{ MP_ROM_QSTR(MP_QSTR_set_mode), MP_ROM_PTR(&gpio_set_mode) },
+	{ MP_ROM_QSTR(MP_QSTR_get_mode), MP_ROM_PTR(&gpio_get_mode) },
+	{ MP_ROM_QSTR(MP_QSTR_read), MP_ROM_PTR(&gpio_read) },
+	{ MP_ROM_QSTR(MP_QSTR_write), MP_ROM_PTR(&gpio_write) },
+	{ MP_ROM_QSTR(MP_QSTR_WRISTBAND_1),
+	  MP_OBJ_NEW_SMALL_INT(GPIO_WRISTBAND_1) },
+	{ MP_ROM_QSTR(MP_QSTR_WRISTBAND_2),
+	  MP_OBJ_NEW_SMALL_INT(GPIO_WRISTBAND_2) },
+	{ MP_ROM_QSTR(MP_QSTR_WRISTBAND_3),
+	  MP_OBJ_NEW_SMALL_INT(GPIO_WRISTBAND_3) },
+	{ MP_ROM_QSTR(MP_QSTR_WRISTBAND_4),
+	  MP_OBJ_NEW_SMALL_INT(GPIO_WRISTBAND_4) },
+	{ MP_ROM_QSTR(MP_QSTR_mode), MP_ROM_PTR(&gpio_module_modes) },
+};
+static MP_DEFINE_CONST_DICT(gpio_module_globals, gpio_module_globals_table);
+
+const mp_obj_module_t gpio_module = {
+	.base    = { &mp_type_module },
+	.globals = (mp_obj_dict_t *)&gpio_module_globals,
+};
+
+/* This is a special macro that will make MicroPython aware of this module */
+/* clang-format off */
+MP_REGISTER_MODULE(MP_QSTR_gpio, gpio_module, MODULE_GPIO_ENABLED);
diff --git a/pycardium/modules/qstrdefs.h b/pycardium/modules/qstrdefs.h
index e1bb0a02..49ad16f4 100644
--- a/pycardium/modules/qstrdefs.h
+++ b/pycardium/modules/qstrdefs.h
@@ -100,3 +100,18 @@ Q(exit)
 Q(exec)
 Q(listdir)
 Q(unlink)
+
+/* gpio */
+Q(gpio)
+Q(set_mode)
+Q(get_mode)
+Q(read)
+Q(write)
+Q(WRISTBAND_1)
+Q(WRISTBAND_2)
+Q(WRISTBAND_3)
+Q(WRISTBAND_4)
+Q(INPUT)
+Q(OUTPUT)
+Q(PULL_UP)
+Q(PULL_DOWN)
diff --git a/pycardium/mpconfigport.h b/pycardium/mpconfigport.h
index 00430b2e..0708ebe5 100644
--- a/pycardium/mpconfigport.h
+++ b/pycardium/mpconfigport.h
@@ -41,6 +41,7 @@
 /* Modules */
 #define MODULE_BUTTONS_ENABLED              (1)
 #define MODULE_DISPLAY_ENABLED              (1)
+#define MODULE_GPIO_ENABLED                 (1)
 #define MODULE_INTERRUPT_ENABLED            (1)
 #define MODULE_LEDS_ENABLED                 (1)
 #define MODULE_LIGHT_SENSOR_ENABLED         (1)
-- 
GitLab