HP85 GPIB Disk Emulator  1.0
HP85GPIBDiskEmulator
All Data Structures Files Functions Variables Typedefs Enumerations Enumerator Macros Pages
i2c.c
Go to the documentation of this file.
1 
19 #include "user_config.h"
20 
23 
27 typedef int8_t (*i2c_callback_t)(void);
29 
31 i2c_task_t i2c_task = { 0,0,0,0 };
32 
34 
35 
40 uint8_t i2c_check_op(int8_t index)
41 {
42  if(index < 0 || index >= I2C_OPS)
43  {
44  printf("I2C op[INDEX %d >= %d]\n",(int) index, I2C_OPS);
45  return(0);
46  }
47  if(i2c_task_op[index] == NULL)
48  {
49  return(0);
50  }
51  if(i2c_task_op[index]->buf == NULL)
52  {
53  printf("I2C op[%d]->buf == NULL\n",(int) index);
54  return(0);
55  }
56  if(i2c_task_op[index]->len == 0)
57  {
58  printf("I2C op[%d]->len == 0\n",(int) index);
59  return(0);
60  }
61 
62  return(1);
63 }
64 
70 {
71 
72  int8_t i;
73  uint8_t sreg = SREG;
74 
75  cli();
76 
77  for(i=0;i<I2C_OPS;++i)
78  i2c_task_op[i] = NULL;
79  i2c_task.enable = 0;
80  i2c_task.done = 1;
81  i2c_task.ind = 0;
82  i2c_task.error = 0;
83 
84  SREG = sreg;
85 }
86 
87 
93 {
94  i2c_op_t *o;
95  int8_t i;
96  uint8_t sreg = SREG;
97 
98  cli();
99 
100  for(i=0;i2c_check_op(i);++i)
101  {
102  o = i2c_task_op[i];
103  // We ASSUME the buffer is static and not allocated
104  o->buf = NULL;
105  o->done = 0;
106  o->enable = 0;
107  o->flags = 0;
108  o->timeout = 0;
109  o->ind = 0;
110  safefree(o);
111  i2c_task_op[i] = NULL;
112  }
113 
114 // disable interrupts
115  TWCR = 0;
116 
117 // disable slave mode
118  TWAR = 0;
119 
120  // Reset status
121  TWSR &= ~(_BV(TWPS0) | _BV(TWPS1));
122 
123  SREG = sreg;
124 }
125 
126 
127 
135 i2c_op_t *i2c_task_op_add(uint8_t address, uint8_t mode, uint8_t *buf, uint8_t len)
136 {
137  uint8_t sreg = SREG;
138  i2c_op_t *o;
139 
140 
141  o = safecalloc(1,sizeof(i2c_op_t));
142  if(o == NULL)
143  return(o);
144 
145  cli();
146 
147  o->enable = 0; // NOT enabled
148  o->done = 0;
149  o->address = (address << 1) | (mode & 1);
150  o->timeout = I2C_TIMEOUT;
151  o->flags = 0;
152  o->len = len;
153  o->ind = 0;
154  o->buf = buf;
155 
156  SREG = sreg;
157  return(o);
158 }
159 
164 {
165  i2c_op_t *o;
166 
167  // NOTE: we are in an ISR do not disable other interrupts
168 
169  if(i2c_task.enable)
170  {
171  // Save state of LAST operation
173  {
174  o = i2c_task_op[i2c_task.ind];
175  // Save state in last opperation
176  *o = i2c;
177  if(o->flags)
178  i2c_task.error = 1;
179  }
180 
181  if(i2c_check_op(i2c_task.ind+1) )
182  {
183  i2c_task.ind++;
184 
185  o = i2c_task_op[i2c_task.ind];
186 
187  if( o->enable == 1 || o->done == 0)
188  {
189  o->timeout = I2C_TIMEOUT;
190  o->flags = 0;
191  o->ind = 0;
192  o->enable = 1;
193  o->done = 0;
194 
195  i2c = *o;
196 
197  i2c_send_start();
198 
199  return(1);
200  }
201  }
202  }
203  // program error
204  // Disable TASK
205  i2c_task.enable = 0;
206  i2c_task.done = 1;
207 
208  // I2C disable
209  i2c.done = 1;
210  i2c.enable = 0;
211  i2c.flags = 0;
212  i2c.ind = 0;
213 
214  i2c_send_stop();
215 
216  return(0);
217 }
218 
219 
220 
221 
226 {
227  uint8_t sreg = SREG;
228  i2c_op_t *o;
229  uint8_t run = 0;
230  int8_t i;
231 
232 
233  cli();
234  i2c_task.done = 0;
235  i2c_task.error = 0;
236  i2c_task.ind = 0;
237 
238  // re-enable tasks
239  for(i=0;i2c_check_op(i);++i)
240  {
241  o = i2c_task_op[i];
242  o->enable = 1;
243  o->done = 0;
244  o->flags = 0;
245  o->ind = 0;
246  o->timeout = I2C_TIMEOUT;
247  if(!run)
248  {
249  run = 1;
250  i2c = *o;
251  }
252  }
253 
254  if(run)
255  {
256  // TASK callback
258 
259  i2c_task.enable = 1;
260  i2c_task.done = 0;
261 
262  // Reset Status
263  TWSR &= ~(_BV(TWPS0) | _BV(TWPS1));
264  // Start a transactions
265  // TWI Enable
266  // TWI Interrupt Enable
267  // TWI Interrupt Clear
268  // TWI SEND RESTART
269  // TWCR = _BV(TWEN) | _BV(TWIE) | _BV(TWINT) | _BV(TWSTA);
270  // Disable Slave Mode
271 
272  i2c_send_start();
273 
274  TWAR = 0;
275 
276  }
277  else
278  {
279  // FIXME we should notify the user ?
280  // Nothing to DO
281  // User Error
282 
283  // TASK callback
284  i2c_callback = NULL;
285 
286  // TASK Nothing to DO
287  i2c_task.enable = 0;
288  i2c_task.done = 1;
289 
290  // I2C Nothing to do
291  i2c.done = 1;
292  i2c.enable = 0;
293 
294  // TWI Enable
295  // TWI Disable Enable
296  // TWI Interrupt Clear
297  TWCR = _BV(TWEN) | _BV(TWINT);
298  // Reset status
299  TWSR &= ~(_BV(TWPS0) | _BV(TWPS1));
300  // Disable Slave Mode
301  TWAR = 0;
302  }
303  // Disable Slave Mode
304 
305  SREG = sreg;
306 
307 }
308 
309 static int8_t i2c_init_status = 0;
310 
316 void i2c_init(uint32_t speed)
317 {
318  uint8_t sreg = SREG;
319  uint16_t rate = ((F_CPU / speed) - 16) / 2;
320 
321  if(rate & 0xff00)
322  {
323  printf("i2c_init prescale overflow\n");
324  return;
325  }
326 
327  cli();
328 
329  TWBR = rate;
330 
331  i2c.enable = 0; // NOT enabled
332  i2c.done = 1;
333  i2c_task.enable = 0;
334  i2c_task.done = 1;
335 
336  GPIO_PIN_LATCH_HI(SCL); // Pull Up on
337  GPIO_PIN_LATCH_HI(SDA); // Pull Up on
338 
339  // TWI Enable
340  // TWI Interrupt Disable
341  // TWI Interrupt Clear
342  TWCR = _BV(TWEN) | _BV(TWINT);
343 
344  // Disable SLAVE
345  TWAR = 0;
346 
347  // Reset Status
348  TWSR &= ~(_BV(TWPS0) | _BV(TWPS1));
349 
350  if(!i2c_init_status)
351  {
352  int8_t i;
353  for(i=0;i<I2C_OPS;++i)
354  i2c_task_op[i] = NULL;
355  // Enable timer task to monitor timeouts
356  if(set_timers(i2c_timer,1) == -1)
357  printf("i2c_timer init failed\n");
358  i2c_init_status = 1;
359  }
360 
361 /* Restore the status register. */
362  SREG = sreg;
363 }
364 
369 int8_t i2c_ok()
370 {
371  if (i2c.flags)
372  return(0);
373  return(1);
374 }
375 
379 int8_t i2c_done()
380 {
381  delayus(1);
382  if(!i2c.enable || i2c.done )
383  return(1);
384  return(0);
385 }
386 
391 {
392  delayus(1);
393  if(!i2c_task.enable || i2c_task.done )
394  return(1);
395  return(0);
396 }
397 
398 
399 
407 uint8_t i2c_fn(uint8_t address, uint8_t mode, uint8_t *buf, uint8_t len)
408 {
409  uint8_t sreg = SREG;
410 
411  cli();
412 
413  // sign task only
414  i2c_callback = NULL;
415 
416  i2c.enable = 1; // Enabled
417  i2c.done = 0;
418  i2c.address = (address << 1) | (mode & 1);
419  i2c.flags = 0;
420  i2c.len = len;
421  i2c.ind = 0;
422  i2c.buf = buf;
423 
424  // Reset Status
425  TWSR &= ~(_BV(TWPS0) | _BV(TWPS1));
426 
427  // Disable Slave Mode
428  TWAR = 0;
429 
430  SREG = sreg;
431 
432  i2c_send_start();
433 
434 // Debugging
435 #if 0
436  while(i2c.enable && !i2c.done )
437  {
438  printf("timeout: %d\r",(int)i2c.timeout);
439  }
440  printf("\n");
441 #else
442  while(! i2c_done() )
443  ;
444 #endif
445  return( i2c.flags ? 0 : 1);
446 }
447 
452 {
453  i2c.done = 0;
454  i2c.enable = 1;
455  i2c.timeout = I2C_TIMEOUT; // Start timeout timer
456 
457  // Start a transactions
458  // TWI Enable
459  // TWI Interrupt Enable
460  // TWI Interrupt Clear
461  // TWI SEND RESTART
462  TWCR = _BV(TWEN) | _BV(TWIE) | _BV(TWINT) | _BV(TWSTA);
463 }
464 
469 {
470  // All transactions are done
471  i2c.done = 1;
472  i2c.enable = 0;
473 
474  // We are DONE
475  // TWI Enable
476  // TWI Interrupt Disable
477  // TWI Interrupt Clear
478  // TWI SEND STOP
479  TWCR = _BV(TWEN) | _BV(TWINT) | _BV(TWSTO);
480  delayus(10);
481 
482 }
483 
484 
489 static void i2c_next()
490 {
491 
492  // IF we have an i2c_callback() function then
493  // it must save status and reset i2c structure for next operation
494 
495  if(i2c_callback)
496  i2c_callback();
497  else
498  i2c_send_stop();
499 }
500 
503 void i2c_timer()
504 {
505  uint8_t sreg = SREG;
506 
507  if(i2c.enable || !i2c.done )
508  {
509  if(i2c.timeout)
510  i2c.timeout--;
511  }
512  SREG = sreg;
513 }
514 
515 
519 ISR(TWI_vect)
520 {
521  // FYI: reading TWSR clears the status
522  // twi.h defines TW_STATUS
523  // #define TW_STATUS (TWSR & TW_STATUS_MASK)
524  uint8_t status = TW_STATUS;
525 
526  // Are we Enabled to Receive/Send ?
527  // Are we Done ?
528  // Program errors - these should not happen
529  if(!i2c.enable || i2c.done || !i2c.buf || !i2c.len )
530  {
531  i2c_send_stop();
532  return;
533  }
534 
535  // TUMEOUT ? STOP everything
536  if(i2c.timeout == 0)
537  {
538  i2c.flags |= (I2C_OP_TIMEOUT);
539  i2c_task.enable = 0;
540  i2c_task.done = 1;
541  i2c_task.error = 1;
542  i2c_send_stop();
543  return;
544  }
545 
546  // Master Receiver mode
547  switch (status)
548  {
549  case TW_START: // START has been transmitted
550  case TW_REP_START: // RE-START has been transmitted
551  i2c.ind = 0;
552  TWDR = i2c.address;
553  // TWI Enable
554  // TWI Interrupt Enable
555  // TWI Interrupt Clear
556  TWCR = _BV(TWEN) | _BV(TWIE) | _BV(TWINT);
557  break;
558 
559 
560  case TW_MT_SLA_ACK: // SLA+W trasnmitted and ACK received
561  case TW_MT_DATA_ACK:// Data trasnmitted and ACK received
562  if (i2c.ind < i2c.len)
563  {
564  TWDR = i2c.buf[i2c.ind++];
565  // TWI Enable
566  // TWI Interrupt Enable
567  // TWI Interrupt Clear
568  // SEND ACK after receive
569  TWCR = _BV(TWEN) | _BV(TWIE) | _BV(TWINT);
570  }
571  else
572  {
573  // Done
574  i2c.done = 1;
575  i2c.enable = 0;
576  i2c_next();
577  }
578  break;
579 
580  case TW_MR_DATA_ACK: // Data received ACK transmitted
581  if(i2c.ind < i2c.len)
582  i2c.buf[i2c.ind++] = TWDR;
583  case TW_MR_SLA_ACK: // SLA+R transmitted ACK received
584  if ((i2c.ind+1) < i2c.len)
585  {
586  // TWI Enable
587  // TWI Interrupt Enable
588  // TWI Interrupt Clear
589  // SEND ACK after receive
590  TWCR = _BV(TWEN) | _BV(TWIE) | _BV(TWINT) | _BV(TWEA);
591  }
592  else
593  {
594  // LAST BYTE
595  // TWI Enable
596  // TWI Interrupt Enable
597  // TWI Interrupt Clear
598  // SEND NACK after receive
599  TWCR = _BV(TWEN) | _BV(TWIE) | _BV(TWINT);
600  }
601  break;
602 
603  case TW_MR_DATA_NACK: // Data received NACK transmitted
604  if(i2c.ind < i2c.len)
605  i2c.buf[i2c.ind++] = TWDR;
606  i2c.done = 1;
607  i2c.enable = 0;
608  i2c_next();
609  break;
610 
611  // Arbitration lost
612  // NOTE: TW_ARB_LOST == TW_MR_ARB_LOST == TW_MT_ARB_LOST
613  case TW_ARB_LOST:
614  // TWI Enable
615  // TWI Interrupt Enable
616  // TWI Interrupt Clear
617  // TWI SEND RESTART
618  TWCR = _BV(TWEN) | _BV(TWIE) | _BV(TWINT) | _BV(TWSTA);
619  break;
620 
621 // Error cases
622  case TW_MT_SLA_NACK: // SLA+W transmitted NACK received
623  i2c.done = 1;
624  i2c.enable = 0;
626  // ERROR
627  i2c_next();
628  break;
629 
630  case TW_MR_SLA_NACK: // SLA+R transmitted NACK received
631  i2c.done = 1;
632  i2c.enable = 0;
634  // ERROR
635  i2c_next();
636  break;
637 
638  case TW_MT_DATA_NACK: // Data Transmitted NACK received
639  i2c.done = 1;
640  i2c.enable = 0;
642  // ERROR
643  i2c_next();
644  break;
645 
646  default: // Error
647  i2c.done = 1;
648  i2c.enable = 0;
649  // ERROR
651  i2c_next();
652  break;
653  }
654 }
655 
661 {
662  int flags = o->flags;
663 
664  if(flags)
665  {
666  printf(" %s\n", (i2c.done ? "DONE" : "ACTIVE") );
667  if(flags & I2C_OP_TIMEOUT)
668  printf(" OP_TIMEOUT\n");
669  if(flags & I2C_OP_LEN)
670  printf(" OP_LEN\n");
671  if(flags & I2C_OP_ERROR)
672  printf(" OP_ERROR\n");
673  if(flags & I2C_TW_MR_SLA_NACK)
674  printf(" TW_MR_SLA_NACK\n");
675  if(flags & I2C_TW_MT_SLA_NACK)
676  printf(" TW_MT_SLA_NACK\n");
677  if(flags & I2C_TW_MT_DATA_NACK)
678  printf(" TW_MT_DATA_NACK\n");
679  printf("\n");
680  }
681 }
682 
685 {
686  int8_t i;
687  printf("i2c_task.done: %d\n", (int) i2c_task.done);
688  printf("i2c_task.error: %d\n", (int) i2c_task.error);
689  for(i=0;i2c_task_op[i] && i < I2C_OPS;++i)
690  {
691  printf("task: %d\n", (int) i);
693  printf("\n");
694  }
695 }
i2c_op_t::buf
uint8_t * buf
Definition: i2c.h:49
i2c_task_op
i2c_op_t * i2c_task_op[I2C_OPS]
I2C task list.
Definition: i2c.c:33
i2c_task_done
int8_t i2c_task_done()
Are all i2c_task_op[] pointers done sending/receiving ?
Definition: i2c.c:390
printf
MEMSPACE int printf(const char *format,...)
i2c_check_op
uint8_t i2c_check_op(int8_t index)
Check if I2C structure pointer is valid.
Definition: i2c.c:40
i2c_callback
i2c_callback_t i2c_callback
Definition: i2c.c:28
safecalloc
void * safecalloc(int size, int elements)
Safe Alloc - Display Error message if Calloc fails.
Definition: ram.c:122
i2c
i2c_op_t i2c
I2C interrupt state registers.
Definition: i2c.c:22
i2c_display_task_errors
void i2c_display_task_errors()
Display any task errors.
Definition: i2c.c:684
i2c_done
int8_t i2c_done()
Is i2c structure done sending/receiving ?
Definition: i2c.c:379
i2c_init
void i2c_init(uint32_t speed)
I2C initialize Clear all i2c_task_op[] pointers and disables I2C tasks.
Definition: i2c.c:316
i2c_task_t
Definition: i2c.h:52
i2c_ok
int8_t i2c_ok()
check if I2C trasnaction detected an error
Definition: i2c.c:369
I2C_TIMEOUT
#define I2C_TIMEOUT
Definition: i2c.h:30
i2c_task_t::done
uint8_t done
Definition: i2c.h:55
safefree
void safefree(void *p)
Safe free - Only free a pointer if it is in malloc memory range.
Definition: ram.c:158
i2c_timer
void i2c_timer()
I2C timer task - check for I2C operation timeouts.
Definition: i2c.c:503
I2C_OPS
#define I2C_OPS
Definition: i2c.h:28
I2C_TW_MR_SLA_NACK
#define I2C_TW_MR_SLA_NACK
Definition: i2c.h:36
i2c_task_t::ind
int8_t ind
Definition: i2c.h:54
ISR
ISR(TWI_vect)
I2C ISR for send/receive.
Definition: i2c.c:519
i2c_op_t::enable
uint8_t enable
Definition: i2c.h:43
I2C_BUS_ERROR
#define I2C_BUS_ERROR
Definition: i2c.h:38
i2c_op_t::timeout
uint16_t timeout
Definition: i2c.h:46
i2c_task_op_add
i2c_op_t * i2c_task_op_add(uint8_t address, uint8_t mode, uint8_t *buf, uint8_t len)
I2C setup new OP but do not run it yet.
Definition: i2c.c:135
i2c_task_init
void i2c_task_init()
Initialize I2C task op list NOTE: We ASSUME all i2c_task_op[].buf are statically allocated.
Definition: i2c.c:69
i2c_init_status
static int8_t i2c_init_status
Definition: i2c.c:309
NULL
#define NULL
Definition: user_config.h:85
i2c_op_t::ind
int8_t ind
Definition: i2c.h:48
i2c_task_run
void i2c_task_run()
Run all valid i2c_task_op[] tasks.
Definition: i2c.c:225
i2c_op_t::address
uint8_t address
Definition: i2c.h:42
I2C_OP_TIMEOUT
#define I2C_OP_TIMEOUT
Definition: i2c.h:32
user_config.h
Master Include for FatFs, RTC, Timers AVR8 - Part of HP85 disk emulator.
i2c_fn
uint8_t i2c_fn(uint8_t address, uint8_t mode, uint8_t *buf, uint8_t len)
I2C setup new OP but do not run it yet.
Definition: i2c.c:407
i2c_task_t::error
uint8_t error
Definition: i2c.h:56
i2c_task_next_op
int8_t i2c_task_next_op()
I2C task ISR callback function.
Definition: i2c.c:163
i2c_send_start
void i2c_send_start()
Send I2C START and enable interrupts.
Definition: i2c.c:451
i2c_op_t::flags
uint8_t flags
Definition: i2c.h:45
i2c_next
static void i2c_next()
Is there anything else to send ? Called after I2C transaction finishes.
Definition: i2c.c:489
i2c_task_free_ops
void i2c_task_free_ops()
Free all allocated memory for i2c_[] pointers NOTE: We ASSUME all i2c_task_op[].buf are statically al...
Definition: i2c.c:92
I2C_OP_ERROR
#define I2C_OP_ERROR
Definition: i2c.h:34
i2c_op_t
Definition: i2c.h:40
i2c_op_t::done
uint8_t done
Definition: i2c.h:44
i2c_send_stop
void i2c_send_stop()
Send I2C STOP and disable interrupts.
Definition: i2c.c:468
I2C_TW_MT_SLA_NACK
#define I2C_TW_MT_SLA_NACK
Definition: i2c.h:35
i2c_task
i2c_task_t i2c_task
I2C task state.
Definition: i2c.c:31
i2c_callback_t
int8_t(* i2c_callback_t)(void)
I2C task list @ brief I2C callback function Used when automattically sending several transactions.
Definition: i2c.c:27
set_timers
MEMSPACE int set_timers(void(*handler)(void), int timer __attribute__((unused)))
Install a user timer task.
Definition: timer.c:59
delayus
void delayus(uint32_t us)
Delay microseconds using AVR acr-libc _delay_us() function.
Definition: delay.c:26
i2c_op_t::len
int8_t len
Definition: i2c.h:47
i2c_task_t::enable
uint8_t enable
Definition: i2c.h:53
TW_ARB_LOST
#define TW_ARB_LOST
Definition: i2c.h:25
i2c_print_error
void i2c_print_error(i2c_op_t *o)
Display Errors for i2c_task_op[index].
Definition: i2c.c:660
I2C_OP_LEN
#define I2C_OP_LEN
Definition: i2c.h:33
I2C_TW_MT_DATA_NACK
#define I2C_TW_MT_DATA_NACK
Definition: i2c.h:37