You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
246 lines
5.5 KiB
246 lines
5.5 KiB
/** @file ir_grab3.c
|
|
@author M. P. Hayes, UCECE
|
|
@date 24 August 2009
|
|
@brief Test program for IR serial communnications.
|
|
|
|
@defgroup ir_grab3 Test program for IR serial communications.
|
|
*/
|
|
|
|
#include "system.h"
|
|
#include "navswitch.h"
|
|
#include "button.h"
|
|
#include "tinygl.h"
|
|
#include "pacer.h"
|
|
#include "ir.h"
|
|
#include "led.h"
|
|
#include "delay.h"
|
|
#include "eeprom.h"
|
|
#include "../fonts/font5x7_1.h"
|
|
#include <string.h>
|
|
|
|
|
|
/* Define polling rate in Hz. */
|
|
#define LOOP_RATE 2000
|
|
|
|
/* Define text update rate (characters per 10 s). */
|
|
#define MESSAGE_RATE 10
|
|
|
|
/* At 2400 baud and 36 kHz modulation, there are 15 cycles per bit.
|
|
With a Sony start code of length 4.5 x 0.6 ms there are 86 cycles. */
|
|
#define COUNT_MAX 250
|
|
|
|
#define CODESEQ_LEN_MAX 25
|
|
|
|
#define CODESEQ_NUM 5
|
|
|
|
#define IR_MODULATION_PERIOD_US (1e6 / IR_MODULATION_FREQ)
|
|
|
|
#define LOOP_TWEAK_US 1.5
|
|
|
|
|
|
typedef enum {STATE_NORMAL, STATE_LEARN, STATE_STORE} state_t;
|
|
|
|
|
|
typedef uint8_t count_t;
|
|
|
|
typedef struct code
|
|
{
|
|
count_t on;
|
|
count_t off;
|
|
} code_t;
|
|
|
|
|
|
static void transmit (code_t *codeseq)
|
|
{
|
|
uint8_t i;
|
|
|
|
for (i = 0; codeseq[i].on != 0; i++)
|
|
{
|
|
ir_tx_set (1, codeseq[i].on);
|
|
ir_tx_set (0, codeseq[i].off);
|
|
}
|
|
}
|
|
|
|
|
|
static uint8_t capture (code_t *codeseq, uint8_t size)
|
|
{
|
|
uint8_t i;
|
|
|
|
for (i = 0; i < size - 1; i++)
|
|
{
|
|
count_t count;
|
|
|
|
for (count = 0; count < COUNT_MAX && ir_rx_get (); count++)
|
|
{
|
|
DELAY_US (IR_MODULATION_PERIOD_US - LOOP_TWEAK_US);
|
|
}
|
|
codeseq[i].on = count;
|
|
|
|
for (count = 0; count < COUNT_MAX && !ir_rx_get (); count++)
|
|
{
|
|
DELAY_US (IR_MODULATION_PERIOD_US - LOOP_TWEAK_US - 1.0);
|
|
}
|
|
codeseq[i].off = count;
|
|
|
|
if (count >= COUNT_MAX)
|
|
{
|
|
/* Mark end of sequence. */
|
|
codeseq[i + 1].on = 0;
|
|
return i + 1;
|
|
}
|
|
}
|
|
/* Sequence is too long. */
|
|
return 0;
|
|
}
|
|
|
|
|
|
static void codeseqs_write (code_t *codeseqs, uint8_t len, uint8_t num)
|
|
{
|
|
eeprom_write (0, codeseqs, sizeof (codeseqs[0]) * len * num);
|
|
}
|
|
|
|
|
|
static void codeseqs_read (code_t *codeseqs, uint8_t len, uint8_t num)
|
|
{
|
|
/* When the EEPROM is erased all the bytes are 0xFF so set to
|
|
sensible defaults. */
|
|
eeprom_read (0, codeseqs, sizeof (codeseqs[0]) * len * num);
|
|
if (codeseqs[0].on == 0xff)
|
|
{
|
|
uint8_t i;
|
|
|
|
for (i = 0; i < num; i++)
|
|
codeseqs[i * len].on = 0;
|
|
|
|
codeseqs_write (codeseqs, len, num);
|
|
}
|
|
}
|
|
|
|
|
|
static int switch_get (void)
|
|
{
|
|
if (navswitch_push_event_p (NAVSWITCH_PUSH))
|
|
return 0;
|
|
else if (navswitch_push_event_p (NAVSWITCH_NORTH))
|
|
return 1;
|
|
else if (navswitch_push_event_p (NAVSWITCH_EAST))
|
|
return 2;
|
|
else if (navswitch_push_event_p (NAVSWITCH_SOUTH))
|
|
return 3;
|
|
else if (navswitch_push_event_p (NAVSWITCH_WEST))
|
|
return 4;
|
|
|
|
return -1;
|
|
}
|
|
|
|
|
|
int main (void)
|
|
{
|
|
state_t state = STATE_NORMAL;
|
|
code_t codeseq[CODESEQ_LEN_MAX];
|
|
code_t codeseqs[CODESEQ_LEN_MAX * CODESEQ_NUM];
|
|
int seq;
|
|
static const char *strings[] = {"P", "N", "E", "S", "W"};
|
|
|
|
codeseqs_read (codeseqs, CODESEQ_LEN_MAX, CODESEQ_NUM);
|
|
|
|
system_init ();
|
|
tinygl_init (LOOP_RATE);
|
|
tinygl_font_set (&font5x7_1);
|
|
tinygl_text_speed_set (MESSAGE_RATE);
|
|
|
|
navswitch_init ();
|
|
button_init ();
|
|
led_init ();
|
|
ir_init ();
|
|
|
|
pacer_init (LOOP_RATE);
|
|
|
|
tinygl_text ("X");
|
|
led_set (LED1, 0);
|
|
|
|
while (1)
|
|
{
|
|
/* Wait for next tick. */
|
|
pacer_wait ();
|
|
|
|
switch (state)
|
|
{
|
|
case STATE_NORMAL:
|
|
|
|
tinygl_update ();
|
|
navswitch_update ();
|
|
button_update ();
|
|
|
|
seq = switch_get ();
|
|
|
|
if (seq != -1)
|
|
{
|
|
tinygl_text (strings[seq]);
|
|
transmit (&codeseqs[seq * CODESEQ_LEN_MAX]);
|
|
}
|
|
|
|
if (button_push_event_p (BUTTON1))
|
|
{
|
|
led_set (LED1, 1);
|
|
state = STATE_LEARN;
|
|
}
|
|
break;
|
|
|
|
case STATE_LEARN:
|
|
|
|
while (1)
|
|
{
|
|
/* Loop as fast as possible so that capture start of IR transmission
|
|
as soon as possible. */
|
|
button_update ();
|
|
|
|
if (ir_rx_get ())
|
|
{
|
|
if (capture (codeseq, CODESEQ_LEN_MAX))
|
|
{
|
|
tinygl_text ("?");
|
|
led_set (LED1, 1);
|
|
state = STATE_STORE;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (button_push_event_p (BUTTON1))
|
|
{
|
|
led_set (LED1, 0);
|
|
state = STATE_NORMAL;
|
|
break;
|
|
}
|
|
}
|
|
break;
|
|
|
|
case STATE_STORE:
|
|
|
|
tinygl_update ();
|
|
navswitch_update ();
|
|
button_update ();
|
|
|
|
seq = switch_get ();
|
|
|
|
if (seq != -1)
|
|
{
|
|
memcpy (&codeseqs[seq * CODESEQ_LEN_MAX], codeseq, sizeof (codeseq));
|
|
codeseqs_write (codeseqs, CODESEQ_LEN_MAX, CODESEQ_NUM);
|
|
tinygl_text (strings[seq]);
|
|
led_set (LED1, 0);
|
|
state = STATE_NORMAL;
|
|
}
|
|
|
|
if (button_push_event_p (BUTTON1))
|
|
{
|
|
led_set (LED1, 0);
|
|
state = STATE_NORMAL;
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|