Subversion Repositories Projects

Rev

Rev 346 | Rev 389 | Go to most recent revision | Details | Compare with Previous | Last modification | View Log | RSS feed

Rev Author Line No. Line
331 cascade 1
/****************************************************************************
2
 *   Copyright (C) 2009 by Claas Anders "CaScAdE" Rathje                    *
3
 *   admiralcascade@gmail.com                                               *
4
 *   Project-URL: http://www.mylifesucks.de/oss/c-osd/                      *
5
 *                                                                          *
6
 *   This program is free software; you can redistribute it and/or modify   *
7
 *   it under the terms of the GNU General Public License as published by   *
8
 *   the Free Software Foundation; either version 2 of the License.         *
9
 *                                                                          *
10
 *   This program is distributed in the hope that it will be useful,        *
11
 *   but WITHOUT ANY WARRANTY; without even the implied warranty of         *
12
 *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the          *
13
 *   GNU General Public License for more details.                           *
14
 *                                                                          *
15
 *   You should have received a copy of the GNU General Public License      *
16
 *   along with this program; if not, write to the                          *
17
 *   Free Software Foundation, Inc.,                                        *
18
 *   59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.              *
19
 ****************************************************************************/
20
 
346 cascade 21
#include <avr/io.h>
22
#include <avr/interrupt.h>
23
#include <util/delay.h>
24
#include "main.h"
331 cascade 25
#include "usart1.h"
26
 
346 cascade 27
volatile uint8_t rxd_buffer_locked = 0;
28
volatile uint8_t rxd_buffer[RXD_BUFFER_LEN];
29
volatile uint8_t txd_buffer[TXD_BUFFER_LEN];
30
volatile uint8_t ReceivedBytes = 0;
31
volatile uint8_t *pRxData = 0;
32
volatile uint8_t RxDataLen = 0;
33
 
331 cascade 34
/* ##########################################################################
35
 * USART stuff
36
 * ##########################################################################*/
37
 
38
/**
39
 * init usart1
40
 */
41
void usart1_init() {
42
    UBRR1H = ((F_CPU / (16UL * baud)) - 1) >> 8;
43
    UBRR1L = (F_CPU / (16UL * baud)) - 1;
44
 
45
    // Enable receiver and transmitter; enable RX interrupt
46
    UCSR1B = (1 << RXEN1) | (1 << TXEN1) | (1 << RXCIE1);
47
 
48
    //asynchronous 8N1
49
    UCSR1C = (1 << URSEL1) | (3 << UCSZ10);
50
}
51
 
52
/**
53
 * disable the txd pin of usart1
54
 */
55
void usart1_DisableTXD(void) {
387 cascade 56
        //UCSR1B &= ~(1 << TXCIE1); // disable TX-Interrupt
57
        UCSR1B &= ~(1 << TXEN1); // disable TX in USART
331 cascade 58
        DDRB &= ~(1<<DDB3); // set TXD pin as input
59
        PORTB &= ~(1 << PORTB3);
60
}
61
 
62
/**
63
 * enable the txd pin of usart1
64
 */
65
void usart1_EnableTXD(void) {
66
        DDRB |= (1<<DDB3); // set TXD pin as output
67
        PORTB &= ~(1 << PORTB3);
68
        UCSR1B |= (1 << TXEN1); // enable TX in USART
387 cascade 69
        //UCSR1B |= (1 << TXCIE1); // enable TX-Interrupt
331 cascade 70
}
71
 
72
 
73
/**
74
 * send a single <character> through usart1
75
 */
76
void usart1_putc(unsigned char character) {
77
    // wait until UDR ready
78
    while (!(UCSR1A & (1 << UDRE1)));
79
    UDR1 = character;
80
}
81
 
82
/**
83
 * send a <string> throught usart1
84
 */
85
void usart1_puts(char *s) {
86
    while (*s) {
87
        usart1_putc(*s);
88
        s++;
89
    }
90
}
91
 
92
/**
387 cascade 93
 * transmit interrupt handler
94
 * unused
95
 */
96
SIGNAL(SIG_USART1_DATA) {
97
}
98
 
99
/**
331 cascade 100
 * receive data through usart1
101
 * portions taken and adapted from
102
 * http://svn.mikrokopter.de/mikrowebsvn/filedetails.php?repname=FlightCtrl&path=%2Fbranches%2FV0.72p+Code+Redesign+killagreg%2Fuart0.c
103
 */
104
ISR(SIG_USART1_RECV) {
105
    if (rxd_buffer_locked) return; // if rxd buffer is locked immediately return
106
    LED1_ON
107
            static uint16_t crc;
108
    static uint8_t ptr_rxd_buffer = 0;
109
    uint8_t crc1, crc2;
110
    uint8_t c;
111
 
112
    c = UDR1; // catch the received byte
113
 
114
    // the rxd buffer is unlocked
115
    if ((ptr_rxd_buffer == 0) && (c == '#')) // if rxd buffer is empty and syncronisation character is received
116
    {
117
                /*
118
                // skip other datasets
119
        if (ptr_rxd_buffer == 2 && rxd_buffer[ptr_rxd_buffer] != 'O') {
120
                        ptr_rxd_buffer = 0; // reset rxd buffer
121
                rxd_buffer_locked = 0; // unlock rxd buffer
122
                }*/
123
                rxd_buffer[ptr_rxd_buffer++] = c; // copy 1st byte to buffer
124
        crc = c; // init crc
125
    } else if (ptr_rxd_buffer < RXD_BUFFER_LEN) // collect incomming bytes
126
    {
127
        if (c != '\r') // no termination character
128
        {
129
            rxd_buffer[ptr_rxd_buffer++] = c; // copy byte to rxd buffer
130
            crc += c; // update crc
131
        } else // termination character was received
132
        {
133
            // the last 2 bytes are no subject for checksum calculation
134
            // they are the checksum itself
135
            crc -= rxd_buffer[ptr_rxd_buffer - 2];
136
            crc -= rxd_buffer[ptr_rxd_buffer - 1];
137
            // calculate checksum from transmitted data
138
            crc %= 4096;
139
            crc1 = '=' + crc / 64;
140
            crc2 = '=' + crc % 64;
141
            // compare checksum to transmitted checksum bytes
142
            if ((crc1 == rxd_buffer[ptr_rxd_buffer - 2]) && (crc2 == rxd_buffer[ptr_rxd_buffer - 1])) { // checksum valid
143
                rxd_buffer[ptr_rxd_buffer] = '\r'; // set termination character
144
                ReceivedBytes = ptr_rxd_buffer + 1; // store number of received bytes
145
                rxd_buffer_locked = 1; // lock the rxd buffer
146
            } else { // checksum invalid
147
                rxd_buffer_locked = 0; // unlock rxd buffer
148
            }
149
            ptr_rxd_buffer = 0; // reset rxd buffer pointer
150
        }
151
    } else // rxd buffer overrun
152
    {
153
        ptr_rxd_buffer = 0; // reset rxd buffer
154
        rxd_buffer_locked = 0; // unlock rxd buffer
155
    }
156
    LED1_OFF
157
}
158
 
159
/**
160
 * Decode the recevied Buffer
161
 * portions taken and adapted from
162
 * http://svn.mikrokopter.de/mikrowebsvn/filedetails.php?repname=FlightCtrl&path=%2Ftags%2FV0.72p%2Fuart.c
163
 */
164
void Decode64(void) {
165
    uint8_t a, b, c, d;
166
    uint8_t x, y, z;
167
    uint8_t ptrIn = 3;
168
    uint8_t ptrOut = 3;
169
    uint8_t len = ReceivedBytes - 6;
170
 
171
    while (len) {
172
        a = rxd_buffer[ptrIn++] - '=';
173
        b = rxd_buffer[ptrIn++] - '=';
174
        c = rxd_buffer[ptrIn++] - '=';
175
        d = rxd_buffer[ptrIn++] - '=';
176
 
177
        x = (a << 2) | (b >> 4);
178
        y = ((b & 0x0f) << 4) | (c >> 2);
179
        z = ((c & 0x03) << 6) | d;
180
 
181
        if (len--) rxd_buffer[ptrOut++] = x;
182
        else break;
183
        if (len--) rxd_buffer[ptrOut++] = y;
184
        else break;
185
        if (len--) rxd_buffer[ptrOut++] = z;
186
        else break;
187
    }
188
    pRxData = &rxd_buffer[3];
189
    RxDataLen = ptrOut - 3;
190
}
191
 
192
/**
193
 * request Data through USART in special MK format by adding checksum and
194
 * encode data in modified Base64
195
 * portions taken and adapted from
196
 * http://svn.mikrokopter.de/mikrowebsvn/filedetails.php?repname=FlightCtrl&path=%2Ftags%2FV0.72p%2Fuart.c
197
 */
198
void sendMKData(unsigned char cmd, unsigned char addr, unsigned char *snd, unsigned char len) {
199
    unsigned int pt = 0;
200
    unsigned char a, b, c;
201
    unsigned char ptr = 0;
202
 
203
    txd_buffer[pt++] = '#'; // Start-Byte
204
    txd_buffer[pt++] = 'a' + addr; // Adress
205
    txd_buffer[pt++] = cmd; // Command
206
    while (len) {
207
        if (len) {
208
            a = snd[ptr++];
209
            len--;
210
        } else a = 0;
211
        if (len) {
212
            b = snd[ptr++];
213
            len--;
214
        } else b = 0;
215
        if (len) {
216
            c = snd[ptr++];
217
            len--;
218
        } else c = 0;
219
        txd_buffer[pt++] = '=' + (a >> 2);
220
        txd_buffer[pt++] = '=' + (((a & 0x03) << 4) | ((b & 0xf0) >> 4));
221
        txd_buffer[pt++] = '=' + (((b & 0x0f) << 2) | ((c & 0xc0) >> 6));
222
        txd_buffer[pt++] = '=' + (c & 0x3f);
223
    }
224
 
225
    // add crc
226
    unsigned int tmpCRC = 0, i;
227
    for (i = 0; i < pt; i++) {
228
        tmpCRC += txd_buffer[i];
229
    }
230
    tmpCRC %= 4096;
231
    txd_buffer[i++] = '=' + tmpCRC / 64;
232
    txd_buffer[i++] = '=' + tmpCRC % 64;
233
    txd_buffer[i++] = '\r';
234
 
235
    usart1_puts((char*) txd_buffer);
236
}
346 cascade 237
 
238
/**
239
 * short script to directly send a request thorugh usart including en- and disabling it
240
 * where <address> is the address of the receipient, <label> is which data set to request
241
 * and <ms> represents the milliseconds delay between data
242
 */
243
void usart1_request_mk_data(uint8_t address, char label, uint8_t ms) {
244
            // re-enable TXD pin
245
            usart1_EnableTXD();
246
 
247
            unsigned char mstenth = ms/10;
248
                        sendMKData(label, address, &mstenth, 1);
387 cascade 249
 
346 cascade 250
            // disable TXD pin again
251
            usart1_DisableTXD();
252
}