summaryrefslogtreecommitdiff
path: root/src/driver/soft_i2c.cc
diff options
context:
space:
mode:
authorDaniel Friesel <derf@finalrewind.org>2018-01-15 10:16:13 +0100
committerDaniel Friesel <derf@finalrewind.org>2018-01-15 10:16:13 +0100
commit7b8c37d235c77d881db4184a798fc4b72a1fa279 (patch)
tree974caf26a3964f1650285ae80f6c924d29474abd /src/driver/soft_i2c.cc
parentc374e91391d3e58295375f439b15fe888bcdb3bd (diff)
Add Software I2C implementation
Diffstat (limited to 'src/driver/soft_i2c.cc')
-rw-r--r--src/driver/soft_i2c.cc121
1 files changed, 121 insertions, 0 deletions
diff --git a/src/driver/soft_i2c.cc b/src/driver/soft_i2c.cc
new file mode 100644
index 0000000..f8ee052
--- /dev/null
+++ b/src/driver/soft_i2c.cc
@@ -0,0 +1,121 @@
+#include "driver/soft_i2c.h"
+#include "driver/gpio.h"
+
+signed char SoftI2C::setup()
+{
+ gpio.write(sda, 0);
+ gpio.write(scl, 0);
+ return 0;
+}
+
+void SoftI2C::start()
+{
+ gpio.input(sda);
+ gpio.input(scl);
+ //
+ gpio.output(sda);
+ //
+ gpio.output(scl);
+}
+
+void SoftI2C::stop()
+{
+ gpio.output(sda);
+ //
+ gpio.input(scl);
+ //
+ gpio.input(sda);
+}
+
+bool SoftI2C::tx(unsigned char byte)
+{
+ unsigned char got_ack = 0;
+ for (unsigned char i = 0; i <= 8; i++) {
+ if ((byte & 0x80) || (i == 8)) {
+ gpio.input(sda);
+ } else {
+ gpio.output(sda);
+ }
+ byte <<= 1;
+ //
+ gpio.input(scl);
+ //
+ if (i == 8) {
+ if (!gpio.read(sda)) {
+ got_ack = 1;
+ }
+ }
+ gpio.output(scl);
+ //
+ }
+ return got_ack;
+}
+
+unsigned char SoftI2C::rx(bool send_ack)
+{
+ unsigned char byte = 0;
+ gpio.input(sda);
+ for (unsigned char i = 0; i <= 8; i++) {
+ //
+ gpio.input(scl);
+ //
+ if ((i < 8) && gpio.read(sda)) {
+ byte |= 1 << (7 - i);
+ }
+ //
+ gpio.output(scl);
+ //
+ if ((i == 7) && send_ack) {
+ gpio.output(sda);
+ } else if ((i == 8) && send_ack) {
+ gpio.input(sda);
+ }
+ }
+ return byte;
+}
+
+void SoftI2C::scan(unsigned int *results)
+{
+ unsigned char i2caddr;
+ for (unsigned char address = 0; address < 128; address++) {
+
+ i2caddr = (address << 1) | 1;
+
+ start();
+
+ if (tx(i2caddr)) {
+ results[address / (8 * sizeof(unsigned int))] |= 1 << (address % (8 * sizeof(unsigned int)));
+ }
+ }
+ stop();
+}
+
+signed char SoftI2C::xmit(unsigned char address,
+ unsigned char tx_len, unsigned char *tx_buf,
+ unsigned char rx_len, unsigned char *rx_buf)
+{
+ unsigned char i;
+
+ if (tx_len) {
+ start();
+ tx((address << 1) | 0);
+
+ for (i = 0; i < tx_len; i++) {
+ tx(tx_buf[i]);
+ }
+ }
+ if (rx_len) {
+ start();
+ tx((address << 1) | 1);
+
+ for (i = 1; i <= rx_len; i++) {
+ rx_buf[i] = rx((i < rx_len) * 1);
+ }
+ }
+
+ stop();
+
+ return 0;
+}
+
+SoftI2C i2c(GPIO::p1_6, GPIO::p1_7);