diff --git a/os/run/demo.run b/os/run/demo.run
index 253e7bd975..338a8283a9 100644
--- a/os/run/demo.run
+++ b/os/run/demo.run
@@ -10,7 +10,8 @@ set build_components {
drivers/framebuffer drivers/pci drivers/input
}
-lappend_if [have_spec usb] build_components drivers/usb
+lappend_if [have_spec usb] build_components drivers/usb
+lappend_if [have_spec imx53] build_components drivers/gpio
build $build_components
@@ -60,6 +61,13 @@ append_if [have_spec framebuffer] config {
}
+append_if [have_spec imx53] config {
+
+
+
+
+ }
+
append_if [have_spec ps2] config {
@@ -107,6 +115,7 @@ lappend_if [have_spec pci] boot_modules pci_drv
lappend_if [have_spec ps2] boot_modules ps2_drv
lappend_if [have_spec framebuffer] boot_modules fb_drv
lappend_if [have_spec usb] boot_modules usb_drv
+lappend_if [have_spec imx53] boot_modules gpio_drv
build_boot_image $boot_modules
diff --git a/os/src/drivers/gpio/imx53/driver.h b/os/src/drivers/gpio/imx53/driver.h
new file mode 100644
index 0000000000..89581e0d46
--- /dev/null
+++ b/os/src/drivers/gpio/imx53/driver.h
@@ -0,0 +1,407 @@
+/*
+ * \brief Gpio driver for the i.MX53
+ * \author Ivan Loskutov
+ * \author Nikolay Golikov
+ * \date 2012-12-04
+ */
+
+/*
+ * Copyright (C) 2012 Ksys Labs LLC
+ * Copyright (C) 2012 Genode Labs GmbH
+ *
+ * This file is part of the Genode OS framework, which is distributed
+ * under the terms of the GNU General Public License version 2.
+ */
+
+#ifndef _DRIVER_H_
+#define _DRIVER_H_
+
+/* Genode includes */
+#include
+#include
+#include
+#include
+
+/* local includes */
+#include "gpio.h"
+
+
+namespace Gpio {
+
+ using namespace Genode;
+ class Driver;
+}
+
+static int verbose = 0;
+
+class Gpio::Driver
+{
+ public:
+ enum {
+ GPIO1L_IRQ = 50,
+ GPIO1H_IRQ = 51,
+ GPIO2L_IRQ = 52,
+ GPIO2H_IRQ = 53,
+ GPIO3L_IRQ = 54,
+ GPIO3H_IRQ = 55,
+ GPIO4L_IRQ = 56,
+ GPIO4H_IRQ = 57,
+ GPIO5L_IRQ = 103,
+ GPIO5H_IRQ = 104,
+ GPIO6L_IRQ = 105,
+ GPIO6H_IRQ = 106,
+ GPIO7L_IRQ = 107,
+ GPIO7H_IRQ = 108,
+ };
+
+ private:
+
+ struct Timer_delayer : Timer::Connection, Mmio::Delayer
+ {
+ /**
+ * Implementation of 'Delayer' interface
+ */
+ void usleep(unsigned us)
+ {
+ /* polling */
+ if (us == 0)
+ return;
+
+ unsigned ms = us / 1000;
+ if (ms == 0)
+ ms = 1;
+
+ Timer::Connection::msleep(ms);
+ }
+ } _delayer;
+
+ /* memory map */
+ enum {
+ GPIO1_MMIO_BASE = 0x53f84000,
+ GPIO1_MMIO_SIZE = 0x4000,
+
+ GPIO2_MMIO_BASE = 0x53f88000,
+ GPIO2_MMIO_SIZE = 0x4000,
+
+ GPIO3_MMIO_BASE = 0x53f8c000,
+ GPIO3_MMIO_SIZE = 0x4000,
+
+ GPIO4_MMIO_BASE = 0x53f90000,
+ GPIO4_MMIO_SIZE = 0x4000,
+
+ GPIO5_MMIO_BASE = 0x53fdc000,
+ GPIO5_MMIO_SIZE = 0x4000,
+
+ GPIO6_MMIO_BASE = 0x53fe0000,
+ GPIO6_MMIO_SIZE = 0x4000,
+
+ GPIO7_MMIO_BASE = 0x53fe4000,
+ GPIO7_MMIO_SIZE = 0x4000,
+
+ NR_GPIOS = 7,
+ MAX_GPIOS = 224,
+ };
+
+
+ Attached_io_mem_dataspace _gpio1_mmio;
+ Gpio_reg _gpio1;
+ Attached_io_mem_dataspace _gpio2_mmio;
+ Gpio_reg _gpio2;
+ Attached_io_mem_dataspace _gpio3_mmio;
+ Gpio_reg _gpio3;
+ Attached_io_mem_dataspace _gpio4_mmio;
+ Gpio_reg _gpio4;
+ Attached_io_mem_dataspace _gpio5_mmio;
+ Gpio_reg _gpio5;
+ Attached_io_mem_dataspace _gpio6_mmio;
+ Gpio_reg _gpio6;
+ Attached_io_mem_dataspace _gpio7_mmio;
+ Gpio_reg _gpio7;
+
+ Gpio_reg* _gpio_bank[NR_GPIOS];
+
+ bool _irq_enabled[MAX_GPIOS];
+
+ Signal_context_capability _sign[MAX_GPIOS];
+
+ struct Debounce_stat
+ {
+ unsigned int us;
+ bool enable;
+
+ Debounce_stat() : us(0), enable(false) { }
+
+ } _debounce_stat[MAX_GPIOS];
+
+ public:
+
+ Driver();
+
+ bool set_gpio_direction(int gpio, bool is_input);
+ bool set_gpio_dataout(int gpio, bool enable);
+ int get_gpio_datain(int gpio);
+ bool set_gpio_debounce_enable(int gpio, bool enable);
+ bool set_gpio_debouncing_time(int gpio, unsigned int us);
+ bool set_gpio_falling_detect(int gpio, bool enable);
+ bool set_gpio_rising_detect(int gpio, bool enable);
+ bool set_gpio_irq_enable(int gpio, bool enable);
+
+ void register_signal(Signal_context_capability cap, int gpio)
+ {
+ if (!_sign[gpio].valid()) {
+ _sign[gpio] = cap;
+ }
+ }
+
+ void handle_event(int irq_number);
+
+ private:
+ Gpio_reg *_get_gpio_bank(int gpio) { return _gpio_bank[gpio >> 5]; }
+ bool _gpio_valid(int gpio) { return (gpio < MAX_GPIOS) ? true : false; }
+ int _get_gpio_index(int gpio) { return gpio & 0x1f; }
+
+ inline void _irq_signal_send(int gpio)
+ {
+ if (_sign[gpio].valid()) {
+ if (verbose)
+ PDBG("gpio=%d", gpio);
+
+ Signal_transmitter transmitter(_sign[gpio]);
+ transmitter.submit();
+ }
+ }
+
+ inline void _irq_event(int gpio_bank, uint32_t status)
+ {
+ for (int i=0; i<32; i++) {
+ if ((status & (1 << i)) && _irq_enabled[(gpio_bank<<5) + i])
+ _irq_signal_send( (gpio_bank<<5) + i );
+ }
+ }
+
+};
+
+
+Gpio::Driver::Driver()
+:
+ _gpio1_mmio(GPIO1_MMIO_BASE, GPIO1_MMIO_SIZE),
+ _gpio1((addr_t)_gpio1_mmio.local_addr()),
+ _gpio2_mmio(GPIO2_MMIO_BASE, GPIO2_MMIO_SIZE),
+ _gpio2((addr_t)_gpio2_mmio.local_addr()),
+ _gpio3_mmio(GPIO3_MMIO_BASE, GPIO3_MMIO_SIZE),
+ _gpio3((addr_t)_gpio3_mmio.local_addr()),
+ _gpio4_mmio(GPIO4_MMIO_BASE, GPIO4_MMIO_SIZE),
+ _gpio4((addr_t)_gpio4_mmio.local_addr()),
+ _gpio5_mmio(GPIO5_MMIO_BASE, GPIO5_MMIO_SIZE),
+ _gpio5((addr_t)_gpio5_mmio.local_addr()),
+ _gpio6_mmio(GPIO6_MMIO_BASE, GPIO6_MMIO_SIZE),
+ _gpio6((addr_t)_gpio6_mmio.local_addr()),
+ _gpio7_mmio(GPIO7_MMIO_BASE, GPIO7_MMIO_SIZE),
+ _gpio7((addr_t)_gpio7_mmio.local_addr())
+
+{
+ _gpio_bank[0] = &_gpio1;
+ _gpio_bank[1] = &_gpio2;
+ _gpio_bank[2] = &_gpio3;
+ _gpio_bank[3] = &_gpio4;
+ _gpio_bank[4] = &_gpio5;
+ _gpio_bank[5] = &_gpio6;
+ _gpio_bank[6] = &_gpio7;
+
+ for (int i = 0; i < MAX_GPIOS; ++i) {
+ Gpio_reg *gpio_reg = _get_gpio_bank(i);
+ gpio_reg->write(Gpio_reg::Int_conf::HIGH_LEVEL,
+ _get_gpio_index(i));
+ }
+}
+
+bool Gpio::Driver::set_gpio_direction(int gpio, bool is_input)
+{
+ if (verbose) {
+ PDBG("gpio=%d is_input=%d", gpio, is_input);
+ }
+
+ if (!_gpio_valid(gpio))
+ return false;
+
+ Gpio_reg *gpio_reg = _get_gpio_bank(gpio);
+
+ if (is_input)
+ gpio_reg->write(0, _get_gpio_index(gpio));
+ else
+ gpio_reg->write(1, _get_gpio_index(gpio));
+
+ return true;
+}
+
+bool Gpio::Driver::set_gpio_dataout(int gpio, bool enable)
+{
+ if (verbose)
+ PDBG("gpio=%d enable=%d", gpio, enable);
+
+ if (!_gpio_valid(gpio))
+ return false;
+
+ Gpio_reg *gpio_reg = _get_gpio_bank(gpio);
+
+ if (enable)
+ gpio_reg->write(1, _get_gpio_index(gpio));
+ else
+ gpio_reg->write(0, _get_gpio_index(gpio));
+
+ return true;
+}
+
+int Gpio::Driver::get_gpio_datain(int gpio)
+{
+ if (verbose)
+ PDBG("gpio=%d", gpio);
+
+ if (!_gpio_valid(gpio))
+ return -1;
+
+ Gpio_reg *gpio_reg = _get_gpio_bank(gpio);
+
+ if (_debounce_stat[gpio].enable)
+ _delayer.usleep(_debounce_stat[gpio].us);
+
+ return gpio_reg->read(_get_gpio_index(gpio));
+}
+
+bool Gpio::Driver::set_gpio_debounce_enable(int gpio, bool enable)
+{
+ if (verbose)
+ PDBG("gpio=%d enable=%d", gpio, enable);
+
+ if (!_gpio_valid(gpio))
+ return false;
+
+ _debounce_stat[gpio].enable = enable;
+
+ return true;
+}
+
+bool Gpio::Driver::set_gpio_debouncing_time(int gpio, unsigned int us)
+{
+ if (verbose)
+ PDBG("gpio=%d us=%d", gpio, us);
+
+ if (!_gpio_valid(gpio))
+ return false;
+
+ _debounce_stat[gpio].us = us;
+
+ return true;
+}
+
+bool Gpio::Driver::set_gpio_falling_detect(int gpio, bool enable)
+{
+ if (verbose)
+ PDBG("gpio=%d enable=%d", gpio, enable);
+
+ if (!_gpio_valid(gpio))
+ return false;
+
+ Gpio_reg *gpio_reg = _get_gpio_bank(gpio);
+
+ if (enable)
+ gpio_reg->write(Gpio_reg::Int_conf::FAL_EDGE,
+ _get_gpio_index(gpio));
+ else
+ gpio_reg->write(Gpio_reg::Int_conf::HIGH_LEVEL,
+ _get_gpio_index(gpio));
+
+ return true;
+}
+
+bool Gpio::Driver::set_gpio_rising_detect(int gpio, bool enable)
+{
+ if (verbose)
+ PDBG("gpio=%d enable=%d", gpio, enable);
+
+ if (!_gpio_valid(gpio))
+ return false;
+
+ Gpio_reg *gpio_reg = _get_gpio_bank(gpio);
+
+ if (enable)
+ gpio_reg->write(Gpio_reg::Int_conf::RIS_EDGE,
+ _get_gpio_index(gpio));
+ else
+ gpio_reg->write(Gpio_reg::Int_conf::HIGH_LEVEL,
+ _get_gpio_index(gpio));
+
+ return true;
+}
+
+bool Gpio::Driver::set_gpio_irq_enable(int gpio, bool enable)
+{
+ if (verbose)
+ PDBG("gpio=%d enable=%d", gpio, enable);
+
+ if (!_gpio_valid(gpio))
+ return false;
+
+ Gpio_reg *gpio_reg = _get_gpio_bank(gpio);
+
+ if (enable)
+ gpio_reg->write(1, _get_gpio_index(gpio));
+ else
+ gpio_reg->write(0, _get_gpio_index(gpio));
+
+ _irq_enabled[gpio] = enable;
+
+ return true;
+}
+
+void Gpio::Driver::handle_event(int irq_number)
+{
+ if (verbose)
+ PDBG("IRQ #%d\n", irq_number);
+
+ int gpio_bank = 0;
+
+ switch (irq_number) {
+ case GPIO1L_IRQ:
+ case GPIO1H_IRQ:
+ gpio_bank = 0;
+ break;
+ case GPIO2L_IRQ:
+ case GPIO2H_IRQ:
+ gpio_bank = 1;
+ break;
+ case GPIO3L_IRQ:
+ case GPIO3H_IRQ:
+ gpio_bank = 2;
+ break;
+ case GPIO4L_IRQ:
+ case GPIO4H_IRQ:
+ gpio_bank = 3;
+ break;
+ case GPIO5L_IRQ:
+ case GPIO5H_IRQ:
+ gpio_bank = 4;
+ break;
+ case GPIO6L_IRQ:
+ case GPIO6H_IRQ:
+ gpio_bank = 5;
+ break;
+ case GPIO7L_IRQ:
+ case GPIO7H_IRQ:
+ gpio_bank = 6;
+ break;
+ default:
+ PERR("Unknown Irq number!\n");
+ return;
+ }
+
+ int stat = _gpio_bank[gpio_bank]->read();
+
+ if (verbose)
+ PDBG("GPIO1 IRQSTATUS=%08x\n", stat);
+
+ _irq_event(gpio_bank, stat);
+ _gpio_bank[gpio_bank]->write(0xffffffff);
+}
+
+#endif /* _DRIVER_H_ */
diff --git a/os/src/drivers/gpio/imx53/gpio.h b/os/src/drivers/gpio/imx53/gpio.h
new file mode 100644
index 0000000000..153d0df650
--- /dev/null
+++ b/os/src/drivers/gpio/imx53/gpio.h
@@ -0,0 +1,70 @@
+/*
+ * \brief Gpio driver for the i.MX53
+ * \author Nikolay Golikov
+ * \date 2012-12-06
+ */
+
+/*
+ * Copyright (C) 2012 Ksys Labs LLC
+ * Copyright (C) 2012 Genode Labs GmbH
+ *
+ * This file is part of the Genode OS framework, which is distributed
+ * under the terms of the GNU General Public License version 2.
+ */
+
+#ifndef _GPIO_H_
+#define _GPIO_H_
+
+
+/* Genode includes */
+#include
+
+
+struct Gpio_reg : Genode::Mmio
+{
+ Gpio_reg(Genode::addr_t const mmio_base) : Genode::Mmio(mmio_base) { }
+
+ struct Data : Register_array<0x0, 32, 32, 1>
+ {
+ struct Pin : Bitfield<0, 1> { };
+ };
+
+ struct Dir : Register_array<0x4, 32, 32, 1>
+ {
+ struct Pin : Bitfield<0, 1> { };
+ };
+
+ struct Pad_stat : Register_array<0x8, 32, 32, 1>
+ {
+ struct Pin : Bitfield<0, 1> { };
+ };
+
+ struct Int_conf : Register_array<0xc, 32, 32, 2>
+ {
+ enum {
+ LOW_LEVEL,
+ HIGH_LEVEL,
+ RIS_EDGE,
+ FAL_EDGE
+ };
+
+ struct Pin : Bitfield<0, 2> { };
+ };
+
+ struct Int_mask : Register_array<0x14, 32, 32, 1>
+ {
+ struct Pin : Bitfield<0, 1> { };
+ };
+
+ struct Int_stat : Register_array<0x18, 32, 32, 1>
+ {
+ struct Pin : Bitfield<0, 1> { };
+ };
+
+ struct Edge_sel : Register_array<0x1c, 32, 32, 1>
+ {
+ struct Pin : Bitfield<0, 1> { };
+ };
+};
+
+#endif
diff --git a/os/src/drivers/gpio/imx53/irq_handler.h b/os/src/drivers/gpio/imx53/irq_handler.h
new file mode 100644
index 0000000000..0d3a08d547
--- /dev/null
+++ b/os/src/drivers/gpio/imx53/irq_handler.h
@@ -0,0 +1,50 @@
+/*
+ * \brief Gpio irq-handler
+ * \author Ivan Loskutov
+ * \date 2012-06-23
+ */
+
+/*
+ * Copyright (C) 2012 Ksys Labs LLC
+ * Copyright (C) 2012 Genode Labs GmbH
+ *
+ * This file is part of the Genode OS framework, which is distributed
+ * under the terms of the GNU General Public License version 2.
+ */
+
+#ifndef _IRQ_HANDLER_H_
+#define _IRQ_HANDLER_H_
+
+/* Genode includes */
+#include
+#include
+
+/* local includes */
+#include "driver.h"
+
+class Irq_handler : Genode::Thread<4096>
+{
+private:
+
+ int _irq_number;
+ Genode::Irq_connection _irq;
+ Gpio::Driver &_driver;
+
+public:
+
+ Irq_handler(int irq_number, Gpio::Driver &driver)
+ : _irq_number(irq_number), _irq(irq_number), _driver(driver)
+ {
+ start();
+ }
+
+ void entry()
+ {
+ while (1) {
+ _driver.handle_event(_irq_number);
+ _irq.wait_for_irq();
+ }
+ }
+};
+
+#endif /* _IRQ_HANDLER_H_ */
diff --git a/os/src/drivers/gpio/imx53/main.cc b/os/src/drivers/gpio/imx53/main.cc
new file mode 100644
index 0000000000..54ecc58108
--- /dev/null
+++ b/os/src/drivers/gpio/imx53/main.cc
@@ -0,0 +1,227 @@
+/*
+ * \brief Gpio driver for the i.MX53
+ * \author Ivan Loskutov
+ * \author Nikolay Golikov
+ * \date 2012-12-09
+ */
+
+/*
+ * Copyright (C) 2012 Ksys Labs LLC
+ * Copyright (C) 2012 Genode Labs GmbH
+ *
+ * This file is part of the Genode OS framework, which is distributed
+ * under the terms of the GNU General Public License version 2.
+ */
+
+/* Genode includes */
+#include
+#include
+#include
+#include
+#include
+#include
+#include
+#include
+#include
+
+/* local includes */
+#include "driver.h"
+#include "irq_handler.h"
+
+
+namespace Gpio {
+
+ using namespace Genode;
+ class Session_component;
+}
+
+
+class Gpio::Session_component : public Genode::Rpc_object
+{
+ private:
+
+ Driver &_driver;
+
+ Signal_context_capability _read_avail_sigh;
+
+ public:
+ Session_component(Driver &driver)
+ : _driver(driver)
+ {
+ }
+
+ /************************************
+ ** Gpio::Session interface **
+ ************************************/
+
+ void direction_output(int gpio, bool enable)
+ {
+ _driver.set_gpio_dataout(gpio, enable);
+ _driver.set_gpio_direction(gpio, false);
+ }
+
+ void direction_input(int gpio)
+ {
+ _driver.set_gpio_direction(gpio, true);
+ }
+
+ void dataout(int gpio, bool enable)
+ {
+ _driver.set_gpio_dataout(gpio, enable);
+ }
+
+ int datain(int gpio) { return _driver.get_gpio_datain(gpio); }
+
+ void debounce_enable(int gpio, bool enable)
+ {
+ _driver.set_gpio_debounce_enable(gpio, enable);
+ }
+
+ void debouncing_time(int gpio, unsigned int us)
+ {
+ _driver.set_gpio_debouncing_time(gpio, us);
+ }
+
+ void falling_detect(int gpio, bool enable)
+ {
+ _driver.set_gpio_falling_detect(gpio, enable);
+ }
+
+ void rising_detect(int gpio, bool enable)
+ {
+ _driver.set_gpio_rising_detect(gpio, enable);
+ }
+
+ void irq_enable(int gpio, bool enable)
+ {
+ _driver.set_gpio_irq_enable(gpio, enable);
+ }
+
+ void irq_sigh(Signal_context_capability cap, int gpio)
+ {
+ _driver.register_signal(cap, gpio);
+ }
+};
+
+
+int main(int, char **)
+{
+ using namespace Gpio;
+
+ printf("--- i.MX53 gpio driver ---\n");
+
+ Driver driver;
+
+ Irq_handler gpio1l_irq(Driver::GPIO1L_IRQ, driver);
+ Irq_handler gpio1h_irq(Driver::GPIO1H_IRQ, driver);
+ Irq_handler gpio2l_irq(Driver::GPIO2L_IRQ, driver);
+ Irq_handler gpio2h_irq(Driver::GPIO2H_IRQ, driver);
+ Irq_handler gpio3l_irq(Driver::GPIO3L_IRQ, driver);
+ Irq_handler gpio3h_irq(Driver::GPIO3H_IRQ, driver);
+ Irq_handler gpio4l_irq(Driver::GPIO4L_IRQ, driver);
+ Irq_handler gpio4h_irq(Driver::GPIO4H_IRQ, driver);
+ Irq_handler gpio5l_irq(Driver::GPIO5L_IRQ, driver);
+ Irq_handler gpio5h_irq(Driver::GPIO5H_IRQ, driver);
+ Irq_handler gpio6l_irq(Driver::GPIO6L_IRQ, driver);
+ Irq_handler gpio6h_irq(Driver::GPIO6H_IRQ, driver);
+ Irq_handler gpio7l_irq(Driver::GPIO7L_IRQ, driver);
+ Irq_handler gpio7h_irq(Driver::GPIO7H_IRQ, driver);
+
+ /*
+ * Configure GPIO
+ * Example:
+ *
+ *
+ *
+ *
+ *
+ * num - GPIO pin number
+ * mode - input(I) or output(O)
+ * value - output level (1 or 0), only for output mode
+ */
+ try {
+ Genode::Xml_node gpio_node = Genode::config()->xml_node().sub_node("gpio");
+ for (;; gpio_node = gpio_node.next("gpio")) {
+ unsigned num;
+ char mode[2] = {0};
+ unsigned value = 0;
+ bool value_ok;
+
+ do {
+ try {
+ gpio_node.attribute("num").value(&num);
+ }
+ catch (Genode::Xml_node::Nonexistent_attribute)
+ {
+ PERR("Missing \"num\" attribute. Ignore node.");
+ break;
+ }
+
+ try {
+ gpio_node.attribute("mode").value(mode, sizeof(mode));
+ }
+ catch (Genode::Xml_node::Nonexistent_attribute)
+ {
+ PERR("Missing \"mode\" attribute. Ignore node.");
+ break;
+ }
+
+ try {
+ value_ok = gpio_node.attribute("value").value(&value);
+ }
+ catch (Genode::Xml_node::Nonexistent_attribute)
+ {
+ value_ok = false;
+ }
+
+ if (mode[0] == 'O' || mode[0] == 'o') {
+ if (!value_ok) {
+ PERR("Missing \"value\" attribute for Output mode. Ignore node.");
+ break;
+ }
+ if (value > 1) {
+ PERR("Incorrect \"value\" attribute for Output mode. Ignore node.");
+ break;
+ }
+ driver.set_gpio_dataout(num, value);
+ driver.set_gpio_direction(num, false);
+ } else if (mode[0] == 'I' || mode[0] == 'i') {
+ driver.set_gpio_direction(num, true);
+ } else {
+ PERR("Incorrect value of \"mode\" attribute. Ignore node.");
+ break;
+ }
+
+ PDBG("gpio %d mode %s value=%s",
+ num, mode, value_ok ? (value==0 ? "0" : value==1 ? "1" : "error") : "-");
+
+ } while (0);
+ if (gpio_node.is_last("gpio")) break;
+ }
+ }
+ catch (Genode::Xml_node::Nonexistent_sub_node) {
+ PERR("No GPIO config");
+ }
+
+ /*
+ * Initialize server entry point
+ */
+ enum { STACK_SIZE = 4096 };
+ static Cap_connection cap;
+ static Rpc_entrypoint ep(&cap, STACK_SIZE, "gpio_ep");
+
+ /*
+ * Let the entry point serve the gpio session and root interfaces
+ */
+ static Session_component gpio_session(driver);
+ static Static_root gpio_root(ep.manage(&gpio_session));
+
+ /*
+ * Announce service
+ */
+ env()->parent()->announce(ep.manage(&gpio_root));
+
+ Genode::sleep_forever();
+ return 0;
+}
+
diff --git a/os/src/drivers/gpio/imx53/target.mk b/os/src/drivers/gpio/imx53/target.mk
new file mode 100644
index 0000000000..4d2de30c57
--- /dev/null
+++ b/os/src/drivers/gpio/imx53/target.mk
@@ -0,0 +1,8 @@
+TARGET = gpio_drv
+REQUIRES = imx53
+SRC_CC = main.cc
+LIBS = base
+INC_DIR += $(PRG_DIR)
+
+vpath main.cc $(PRG_DIR)
+