Statistics
| Branch: | Tag: | Revision:

amiro-os / components / eeprom / at24.cpp @ 81b48c66

History | View | Annotate | Download (5.809 KB)

1 58fe0e0b Thomas Schöpping
#include <ch.hpp>
2
#include <hal.h>
3
4
#include <type_traits>
5
6
#include <amiro/bus/i2c/I2CDriver.hpp>
7
#include <amiro/bus/i2c/I2CParams.hpp>
8
9
#include <amiro/eeprom/at24.hpp>
10
11
#define AT24_GET_PAGE_OFFSET(_ADDRESS_, _PAGE_SIZE_) (_ADDRESS_ & (_PAGE_SIZE_ - 1u))
12
13
namespace amiro {
14
15
const struct BaseFileStreamVMT eeprom_at24_base_file_stream_methods = {
16
17
    /* .write       */ AT24::write,
18
    /* .read        */ AT24::read,
19
    /* .put         */ 0,
20
    /* .get         */ 0,
21
    /* .close       */ AT24::close,
22
    /* .geterror    */ EEPROM::geterror,
23
    /* .getsize     */ EEPROM::getsize,
24
    /* .getposition */ EEPROM::getposition,
25
    /* .lseek       */ EEPROM::lseek,
26
27
};
28
29
AT24::
30
AT24(size_t size, uint8_t page_size, uint16_t max_t_wr, I2CDriver* i2c_driver) :
31
  EEPROM(&eeprom_at24_base_file_stream_methods, size, page_size, max_t_wr, AT24_SLA, i2c_driver) {
32
33
}
34
35
AT24::
36
~AT24() {
37
38
}
39
40
msg_t
41
AT24::
42
poll_ack(void* instance) {
43
44
  EEPROM* bfs = (EEPROM*) instance;
45
46
  register I2CDriver* bfs_i2c_driver = bfs->i2c_driver;
47
  register msg_t ret_val = RDY_OK;
48
  register i2cflags_t i2c_err_flags;
49
50
  uint8_t dummy_buf[2];
51
52
  I2CRxParams i2c_rxparams;
53
  i2c_rxparams.addr = bfs->i2c_txparams.addr;
54
  i2c_rxparams.rxbuf = dummy_buf;
55
  i2c_rxparams.rxbytes = sizeof(dummy_buf);
56
57
  for (register uint16_t wr_time = 0x0000u; wr_time < bfs->max_t_wr; wr_time++) {
58
59
    bfs_i2c_driver->acquireBus();
60
61
    ret_val = bfs_i2c_driver->masterReceive(&i2c_rxparams);
62
63
    i2c_err_flags = bfs->i2c_driver->getErrors();
64
65
    bfs_i2c_driver->releaseBus();
66
67
    // Wait cycle over
68
    if (ret_val == RDY_OK)
69
      break;
70
71
    // Check for errors, ignoring only I2CD_ACK_FAILURE
72
    if (i2c_err_flags & ~I2CD_ACK_FAILURE) {
73
      break;
74
    }
75
76
    chThdSleepMicroseconds(10);
77
78
  }
79
80
  // This can only be the case when
81
  // * timeout reached
82
  // * i2cError other than I2CD_ACK_FAILURE within time limit
83
  if (ret_val != RDY_OK)
84
    bfs->error = i2c_err_flags;
85
86
  return ret_val;
87
88
}
89
90
91
size_t
92
AT24::
93
write(void* instance, const uint8_t* bp, size_t n) {
94
95
  EEPROM* bfs = (EEPROM*) instance;
96
97
  i2cflags_t i2c_err_flags;
98
  uint8_t i;
99
  uint8_t j;
100
  uint8_t scratchpad[2 + AT24_MAX_PAGE_SIZE];
101
  msg_t ret_val = RDY_OK;
102
  fileoffset_t cur_pos = bfs->position;
103
104
  register size_t tx_bytes;
105
  register const uint8_t* ptr = bp;
106
  register size_t num_bytes;
107
  register uint8_t page_size = bfs->page_size;
108
  register I2CTxParams* bfs_i2c_txparams = &bfs->i2c_txparams;
109
110
  // If no bytes are to be written, shortcut stop
111
  if (!n)
112
    return 0;
113
114
  // Clear error
115
  bfs->error = EEPROM_ERROR_NONE;
116
117
#if HAL_USE_I2C
118
119
  // number of bytes remaining in page starting at cur_pos
120
  // b/c of address rollover withing current page
121
  tx_bytes = AT24_GET_PAGE_OFFSET(cur_pos, page_size);
122
  // adjust for AT24_MAX_PAGE_SIZE < page_size
123
  tx_bytes = AT24_MAX_PAGE_SIZE - AT24_GET_PAGE_OFFSET(tx_bytes, AT24_MAX_PAGE_SIZE);
124
125
  for (num_bytes = n; num_bytes > 0;) {
126
127
    // write address
128
    i = 0;
129
    // Support for 16bit-addressable devices
130
    if (bfs->size > 0x0080u)
131
      scratchpad[i++] = (cur_pos >> 8) & 0xFFu;
132
133
    scratchpad[i++] = cur_pos & 0xFFu;
134
135
    // adjust number of bytes to transfer if end of buffer
136
    if (num_bytes < tx_bytes)
137
      tx_bytes = num_bytes;
138
139
    // copy data
140
    for (j = 0; j < tx_bytes; j++) {
141
      scratchpad[i++] = *ptr++;
142
    }
143
144
    // acknowledge polling
145
146
    // acknowledge polling and
147
    // address device
148
    for (register uint16_t wr_time = 0x0000u; wr_time < bfs->max_t_wr; wr_time++) {
149
150
      bfs->i2c_driver->acquireBus();
151
152
      bfs_i2c_txparams->txbuf = scratchpad;
153
      bfs_i2c_txparams->txbytes = i;
154
      bfs_i2c_txparams->rxbytes = 0;
155
156
      ret_val = bfs->i2c_driver->masterTransmit(bfs_i2c_txparams);
157
158
      i2c_err_flags = bfs->i2c_driver->getErrors();
159
160
      bfs->i2c_driver->releaseBus();
161
162
      // Write cycle over
163
      if (ret_val == RDY_OK)
164
        break;
165
166
      // Check for errors, ignoring only I2CD_ACK_FAILURE
167
      if (i2c_err_flags & ~I2CD_ACK_FAILURE) {
168
        break;
169
      }
170
171
      chThdSleepMicroseconds(10);
172
173
    }
174
175
    // This can only be the case when
176
    // * timeout reached
177
    // * i2cError other than I2CD_ACK_FAILURE within time limit
178
    if (ret_val != RDY_OK)
179
      break;
180
181
    cur_pos += tx_bytes;
182
    num_bytes -= tx_bytes;
183
    tx_bytes = AT24_MAX_PAGE_SIZE;
184
185
  }
186
187
  if (ret_val != RDY_OK)
188
    bfs->error = i2c_err_flags;
189
190
#endif
191
192
  bfs->position = cur_pos;
193
194
  // number of bytes to transfer - number of bytes not transferred
195
  return n - num_bytes;
196
197
}
198
199
size_t
200
AT24::
201
read(void* instance, uint8_t* bp, size_t n) {
202
203
  EEPROM* bfs = (EEPROM*) instance;
204
205
  uint8_t i;
206
  uint8_t scratchpad[2];
207
  msg_t ret_val;
208
  register I2CTxParams* bfs_i2c_txparams = &bfs->i2c_txparams;
209
210
  // If no bytes are to be read, shortcut stop
211
  if (!n)
212
    return 0;
213
214
  // Clear error
215
  bfs->error = EEPROM_ERROR_NONE;
216
217
#if HAL_USE_I2C
218
219
  // Fill address buffer
220
  i = 0;
221
  // Support for 16bit-addressable devices
222
  if (bfs->size > 0x0080u)
223
    scratchpad[i++] = (bfs->position >> 8) & 0xFFu;
224
225
  scratchpad[i++] = bfs->position & 0xFFu;
226
227
  // if device does not answer within timeout, don't read anything
228
  if (poll_ack(bfs) != RDY_OK)
229
    return 0;
230
231
  bfs->i2c_driver->acquireBus();
232
233
  bfs_i2c_txparams->txbuf = scratchpad;
234
  bfs_i2c_txparams->txbytes = i;
235
  bfs_i2c_txparams->rxbuf = bp;
236
  bfs_i2c_txparams->rxbytes = n;
237
238
  // address device
239
  // and read data
240
  ret_val = bfs->i2c_driver->masterTransmit(bfs_i2c_txparams);
241
242
  if (ret_val != RDY_OK)
243
    bfs->error = bfs->i2c_driver->getErrors();
244
245
  bfs->i2c_driver->releaseBus();
246
247
  // We cannot tell where I²C transfer went wrong
248
  // therefore report 0 bytes read
249
  if (ret_val != RDY_OK)
250
    return 0;
251
252
#endif
253
254
  return n;
255
256
}
257
258
/**
259
 * Close EEPROM device.
260
 * \note EEPROMs do not support close semantics.
261
 * \return FILE_OK Closing an EEPROM device will never fail.
262
 */
263
uint32_t
264
AT24::
265
close(void* instance) {
266
267
  (void) instance;
268
  return FILE_OK;
269
270
}
271
272
} /* amiro */