summaryrefslogtreecommitdiff
path: root/fw/fe310/eos/i2c.c
blob: 72e53ae1f1211fcdbace2acbf25b8a59adc89a4e (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
#include <stdlib.h>
#include <stdint.h>

#include "encoding.h"
#include "platform.h"
#include "prci_driver.h"

#include "eos.h"
#include "i2c.h"

void eos_i2c_start(uint32_t baud_rate) {
    eos_i2c_set_baud_rate(baud_rate);

    GPIO_REG(GPIO_PULLUP_EN)    &= ~(1 << IOF_I2C0_SCL);
    GPIO_REG(GPIO_OUTPUT_XOR)   &= ~(1 << IOF_I2C0_SCL);

    GPIO_REG(GPIO_PULLUP_EN)    &= ~(1 << IOF_I2C0_SDA);
    GPIO_REG(GPIO_OUTPUT_XOR)   &= ~(1 << IOF_I2C0_SDA);

    GPIO_REG(GPIO_IOF_SEL)      &= ~IOF0_I2C0_MASK;
    GPIO_REG(GPIO_IOF_EN)       |=  IOF0_I2C0_MASK;
}

void eos_i2c_stop(void) {
    GPIO_REG(GPIO_PULLUP_EN)    |=  (1 << IOF_I2C0_SCL);
    GPIO_REG(GPIO_PULLUP_EN)    |=  (1 << IOF_I2C0_SDA);
    GPIO_REG(GPIO_IOF_EN)       &=  ~IOF0_I2C0_MASK;
}

void eos_i2c_set_baud_rate(uint32_t baud_rate) {
    unsigned long clock_rate = PRCI_get_cpu_freq();
    uint16_t prescaler = (clock_rate / (baud_rate * 5)) - 1;

    I2C0_REGB(I2C_CONTROL) &= ~I2C_CONTROL_EN;
    I2C0_REGB(I2C_PRESCALE_LOW) = prescaler & 0xFF;
    I2C0_REGB(I2C_PRESCALE_HIGH) = (prescaler >> 8) & 0xFF;
    I2C0_REGB(I2C_CONTROL) |= I2C_CONTROL_EN;
}


static int i2c_addr(uint8_t addr, uint8_t rw_flag) {
    I2C0_REGB(I2C_TRANSMIT) = ((addr & 0x7F) << 1) | (rw_flag & 0x1);
    I2C0_REGB(I2C_COMMAND) = I2C_CMD_START | I2C_CMD_WRITE;
    while (I2C0_REGB(I2C_STATUS) & I2C_STATUS_TIP);

    if (I2C0_REGB(I2C_STATUS) & I2C_STATUS_RXACK) return EOS_ERR;
    return EOS_OK;
}

static int i2c_reg(uint8_t reg) {
    I2C0_REGB(I2C_TRANSMIT) = reg;
    I2C0_REGB(I2C_COMMAND) = I2C_CMD_WRITE;
    while (I2C0_REGB(I2C_STATUS) & I2C_STATUS_TIP);

    if (I2C0_REGB(I2C_STATUS) & I2C_STATUS_RXACK) return EOS_ERR;
    return EOS_OK;
}

int eos_i2c_read(uint8_t addr, uint8_t reg, uint8_t *buffer, uint16_t len) {
    uint8_t cmd;
    int rv;
    int i;

    rv = i2c_addr(addr, I2C_WRITE);
    if (rv) return rv;

    rv = i2c_reg(reg);
    if (rv) return rv;

    rv = i2c_addr(addr, I2C_READ);
    if (rv) return rv;

    cmd = I2C_CMD_READ;

    for (i=0; i<len; i++) {
        /* Set NACK to end read, and generate STOP condition */
        if (i == (len - 1)) cmd |= I2C_CMD_ACK | I2C_CMD_STOP;
        I2C0_REGB(I2C_COMMAND) = cmd;
        while (I2C0_REGB(I2C_STATUS) & I2C_STATUS_TIP);
        buffer[i] = I2C0_REGB(I2C_RECEIVE);
    }

    return EOS_OK;
}

int eos_i2c_write(uint8_t addr, uint8_t reg, uint8_t *buffer, uint16_t len) {
    uint8_t cmd;
    int rv;
    int i;

    rv = i2c_addr(addr, I2C_WRITE);
    if (rv) return rv;

    rv = i2c_reg(reg);
    if (rv) return rv;

    cmd = I2C_CMD_WRITE;

    for (i=0; i<len; i++) {
        if (i == (len - 1)) cmd |= I2C_CMD_STOP;
        I2C0_REGB(I2C_TRANSMIT) = buffer[i];
        I2C0_REGB(I2C_COMMAND) = cmd;
        while (I2C0_REGB(I2C_STATUS) & I2C_STATUS_TIP);
        if (I2C0_REGB(I2C_STATUS) & I2C_STATUS_RXACK) return EOS_ERR;
    }

    return EOS_OK;
}

int eos_i2c_read8(uint8_t addr, uint8_t reg, uint8_t *data) {
    return eos_i2c_read(addr, reg, data, 1);
}

int eos_i2c_write8(uint8_t addr, uint8_t reg, uint8_t data) {
    return eos_i2c_write(addr, reg, &data, 1);
}