Subversion Repositories FlightCtrl

Rev

Rev 1645 | Go to most recent revision | Only display areas with differences | Ignore whitespace | Details | Blame | Last modification | View Log | RSS feed

Rev 1645 Rev 1775
1
// ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
1
// ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
2
// + Copyright (c) 04.2007 Holger Buss
2
// + Copyright (c) 04.2007 Holger Buss
3
// + Nur für den privaten Gebrauch
3
// + Nur für den privaten Gebrauch
4
// + www.MikroKopter.com
4
// + www.MikroKopter.com
5
// ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
5
// ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
6
// + Es gilt für das gesamte Projekt (Hardware, Software, Binärfiles, Sourcecode und Dokumentation),
6
// + Es gilt für das gesamte Projekt (Hardware, Software, Binärfiles, Sourcecode und Dokumentation),
7
// + dass eine Nutzung (auch auszugsweise) nur für den privaten (nicht-kommerziellen) Gebrauch zulässig ist.
7
// + dass eine Nutzung (auch auszugsweise) nur für den privaten (nicht-kommerziellen) Gebrauch zulässig ist.
8
// + Sollten direkte oder indirekte kommerzielle Absichten verfolgt werden, ist mit uns (info@mikrokopter.de) Kontakt
8
// + Sollten direkte oder indirekte kommerzielle Absichten verfolgt werden, ist mit uns (info@mikrokopter.de) Kontakt
9
// + bzgl. der Nutzungsbedingungen aufzunehmen.
9
// + bzgl. der Nutzungsbedingungen aufzunehmen.
10
// + Eine kommerzielle Nutzung ist z.B.Verkauf von MikroKoptern, Bestückung und Verkauf von Platinen oder Bausätzen,
10
// + Eine kommerzielle Nutzung ist z.B.Verkauf von MikroKoptern, Bestückung und Verkauf von Platinen oder Bausätzen,
11
// + Verkauf von Luftbildaufnahmen, usw.
11
// + Verkauf von Luftbildaufnahmen, usw.
12
// ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
12
// ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
13
// + Werden Teile des Quellcodes (mit oder ohne Modifikation) weiterverwendet oder veröffentlicht,
13
// + Werden Teile des Quellcodes (mit oder ohne Modifikation) weiterverwendet oder veröffentlicht,
14
// + unterliegen sie auch diesen Nutzungsbedingungen und diese Nutzungsbedingungen incl. Copyright müssen dann beiliegen
14
// + unterliegen sie auch diesen Nutzungsbedingungen und diese Nutzungsbedingungen incl. Copyright müssen dann beiliegen
15
// ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
15
// ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
16
// + Sollte die Software (auch auszugesweise) oder sonstige Informationen des MikroKopter-Projekts
16
// + Sollte die Software (auch auszugesweise) oder sonstige Informationen des MikroKopter-Projekts
17
// + auf anderen Webseiten oder sonstigen Medien veröffentlicht werden, muss unsere Webseite "http://www.mikrokopter.de"
17
// + auf anderen Webseiten oder sonstigen Medien veröffentlicht werden, muss unsere Webseite "http://www.mikrokopter.de"
18
// + eindeutig als Ursprung verlinkt werden
18
// + eindeutig als Ursprung verlinkt werden
19
// ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
19
// ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
20
// + Keine Gewähr auf Fehlerfreiheit, Vollständigkeit oder Funktion
20
// + Keine Gewähr auf Fehlerfreiheit, Vollständigkeit oder Funktion
21
// + Benutzung auf eigene Gefahr
21
// + Benutzung auf eigene Gefahr
22
// + Wir übernehmen keinerlei Haftung für direkte oder indirekte Personen- oder Sachschäden
22
// + Wir übernehmen keinerlei Haftung für direkte oder indirekte Personen- oder Sachschäden
23
// ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
23
// ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
24
// + Die Portierung der Software (oder Teile davon) auf andere Systeme (ausser der Hardware von www.mikrokopter.de) ist nur
24
// + Die Portierung der Software (oder Teile davon) auf andere Systeme (ausser der Hardware von www.mikrokopter.de) ist nur
25
// + mit unserer Zustimmung zulässig
25
// + mit unserer Zustimmung zulässig
26
// ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
26
// ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
27
// + Die Funktion printf_P() unterliegt ihrer eigenen Lizenz und ist hiervon nicht betroffen
27
// + Die Funktion printf_P() unterliegt ihrer eigenen Lizenz und ist hiervon nicht betroffen
28
// ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
28
// ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
29
// + Redistributions of source code (with or without modifications) must retain the above copyright notice,
29
// + Redistributions of source code (with or without modifications) must retain the above copyright notice,
30
// + this list of conditions and the following disclaimer.
30
// + this list of conditions and the following disclaimer.
31
// +   * Neither the name of the copyright holders nor the names of contributors may be used to endorse or promote products derived
31
// +   * Neither the name of the copyright holders nor the names of contributors may be used to endorse or promote products derived
32
// +     from this software without specific prior written permission.
32
// +     from this software without specific prior written permission.
33
// +   * The use of this project (hardware, software, binary files, sources and documentation) is only permittet
33
// +   * The use of this project (hardware, software, binary files, sources and documentation) is only permittet
34
// +     for non-commercial use (directly or indirectly)
34
// +     for non-commercial use (directly or indirectly)
35
// +     Commercial use (for excample: selling of MikroKopters, selling of PCBs, assembly, ...) is only permitted
35
// +     Commercial use (for excample: selling of MikroKopters, selling of PCBs, assembly, ...) is only permitted
36
// +     with our written permission
36
// +     with our written permission
37
// +   * If sources or documentations are redistributet on other webpages, out webpage (http://www.MikroKopter.de) must be
37
// +   * If sources or documentations are redistributet on other webpages, out webpage (http://www.MikroKopter.de) must be
38
// +     clearly linked as origin
38
// +     clearly linked as origin
39
// +   * porting to systems other than hardware from www.mikrokopter.de is not allowed
39
// +   * porting to systems other than hardware from www.mikrokopter.de is not allowed
40
// +  THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
40
// +  THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
41
// +  AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
41
// +  AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
42
// +  IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
42
// +  IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
43
// +  ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
43
// +  ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
44
// +  LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
44
// +  LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
45
// +  CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
45
// +  CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
46
// +  SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
46
// +  SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
47
// +  INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN// +  CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
47
// +  INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN// +  CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
48
// +  ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
48
// +  ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
49
// +  POSSIBILITY OF SUCH DAMAGE.
49
// +  POSSIBILITY OF SUCH DAMAGE.
50
// ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
50
// ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
51
#include <avr/io.h>
51
#include <avr/io.h>
52
#include <avr/interrupt.h>
52
#include <avr/interrupt.h>
53
#include <string.h>
53
#include <string.h>
54
#include <stdlib.h>
54
#include <stdlib.h>
55
#include "spi.h"
55
#include "spi.h"
56
#include "rc.h"
56
#include "rc.h"
57
#include "eeprom.h"
57
#include "eeprom.h"
58
#include "uart0.h"
58
#include "uart0.h"
59
#include "timer0.h"
59
#include "timer0.h"
60
#include "analog.h"
60
#include "analog.h"
61
#include "attitude.h"
61
#include "attitude.h"
62
#include "GPSControl.h"
62
#include "GPSControl.h"
63
#include "flight.h"
63
#include "flight.h"
64
 
64
 
65
//-----------------------------------------
65
//-----------------------------------------
66
#define DDR_SPI DDRB
66
#define DDR_SPI DDRB
67
#define DD_SS   PB4
67
#define DD_SS   PB4
68
#define DD_SCK  PB7
68
#define DD_SCK  PB7
69
#define DD_MOSI PB5
69
#define DD_MOSI PB5
70
#define DD_MISO PB6
70
#define DD_MISO PB6
71
 
71
 
72
// for compatibility reasons gcc3.x <-> gcc4.x
72
// for compatibility reasons gcc3.x <-> gcc4.x
73
#ifndef SPCR
73
#ifndef SPCR
74
#define SPCR   SPCR0
74
#define SPCR   SPCR0
75
#endif
75
#endif
76
#ifndef SPIE
76
#ifndef SPIE
77
#define SPIE   SPIE0
77
#define SPIE   SPIE0
78
#endif
78
#endif
79
#ifndef SPE
79
#ifndef SPE
80
#define SPE    SPE0
80
#define SPE    SPE0
81
#endif
81
#endif
82
#ifndef DORD
82
#ifndef DORD
83
#define DORD   DORD0
83
#define DORD   DORD0
84
#endif
84
#endif
85
#ifndef MSTR
85
#ifndef MSTR
86
#define MSTR   MSTR0
86
#define MSTR   MSTR0
87
#endif
87
#endif
88
#ifndef CPOL
88
#ifndef CPOL
89
#define CPOL   CPOL0
89
#define CPOL   CPOL0
90
#endif
90
#endif
91
#ifndef CPHA
91
#ifndef CPHA
92
#define CPHA   CPHA0
92
#define CPHA   CPHA0
93
#endif
93
#endif
94
#ifndef SPR1
94
#ifndef SPR1
95
#define SPR1   SPR01
95
#define SPR1   SPR01
96
#endif
96
#endif
97
#ifndef SPR0
97
#ifndef SPR0
98
#define SPR0   SPR00
98
#define SPR0   SPR00
99
#endif
99
#endif
100
 
100
 
101
#ifndef SPDR
101
#ifndef SPDR
102
#define SPDR   SPDR0
102
#define SPDR   SPDR0
103
#endif
103
#endif
104
 
104
 
105
#ifndef SPSR
105
#ifndef SPSR
106
#define SPSR   SPSR0
106
#define SPSR   SPSR0
107
#endif
107
#endif
108
#ifndef SPIF
108
#ifndef SPIF
109
#define SPIF   SPIF0
109
#define SPIF   SPIF0
110
#endif
110
#endif
111
#ifndef WCOL
111
#ifndef WCOL
112
#define WCOL   WCOL0
112
#define WCOL   WCOL0
113
#endif
113
#endif
114
//#ifndef SPI2X
114
#ifndef SPI2X
115
#define SPI2X  SPI2X0
115
#define SPI2X  SPI2X0
116
//#endif
116
#endif
117
// -------------------------
117
// -------------------------
118
 
118
 
119
#define SLAVE_SELECT_DDR_PORT   DDRC
119
#define SLAVE_SELECT_DDR_PORT   DDRC
120
#define SLAVE_SELECT_PORT       PORTC
120
#define SLAVE_SELECT_PORT       PORTC
121
#define SPI_SLAVE_SELECT        PC5
121
#define SPI_SLAVE_SELECT        PC5
122
 
122
 
123
#define SPI_TXSYNCBYTE1 0xAA
123
#define SPI_TXSYNCBYTE1 0xAA
124
#define SPI_TXSYNCBYTE2 0x83
124
#define SPI_TXSYNCBYTE2 0x83
125
#define SPI_RXSYNCBYTE1 0x81
125
#define SPI_RXSYNCBYTE1 0x81
126
#define SPI_RXSYNCBYTE2 0x55
126
#define SPI_RXSYNCBYTE2 0x55
127
 
127
 
128
typedef enum
-
 
129
  {
128
typedef enum {
130
    SPI_SYNC1,
129
  SPI_SYNC1,
131
    SPI_SYNC2,
130
  SPI_SYNC2,
132
    SPI_DATA
131
  SPI_DATA
133
  } SPI_RXState_t;
-
 
134
 
132
} SPI_RXState_t;
135
 
133
 
136
// data exchange packets to and From NaviCtrl
134
// data exchange packets to and From NaviCtrl
-
 
135
ToNaviCtrl_t                    toNaviCtrl;
137
ToNaviCtrl_t                    ToNaviCtrl;
-
 
138
FromNaviCtrl_t                  FromNaviCtrl;
136
FromNaviCtrl_t                  fromNaviCtrl;
139
 
137
SPI_VersionInfo_t               SPI_VersionInfo;
140
SPI_VersionInfo_t SPI_VersionInfo;
138
 
141
 
139
 
142
// rx packet buffer
140
// rx packet buffer
143
#define SPI_RXBUFFER_LEN sizeof(FromNaviCtrl)
141
#define SPI_RXBUFFER_LEN sizeof(fromNaviCtrl)
144
uint8_t SPI_RxBuffer[SPI_RXBUFFER_LEN];
142
uint8_t SPI_RxBuffer[SPI_RXBUFFER_LEN];
145
uint8_t SPI_RxBufferIndex = 0;
143
uint8_t SPI_RxBufferIndex = 0;
146
uint8_t SPI_RxBuffer_Request = 0;
144
uint8_t SPI_RxBuffer_Request = 0;
147
 
145
 
148
// tx packet buffer
146
// tx packet buffer
149
#define SPI_TXBUFFER_LEN sizeof(ToNaviCtrl)
147
#define SPI_TXBUFFER_LEN sizeof(toNaviCtrl)
150
uint8_t *SPI_TxBuffer;
148
uint8_t *SPI_TxBuffer;
151
uint8_t SPI_TxBufferIndex = 0;
149
uint8_t SPI_TxBufferIndex = 0;
152
 
150
 
153
uint8_t SPITransferCompleted, SPI_ChkSum;
151
uint8_t SPITransferCompleted, SPI_ChkSum;
154
uint8_t SPI_RxDataValid = 0;
152
uint8_t SPI_RxDataValid = 0;
155
uint8_t NCDataOkay = 0;
153
uint8_t NCDataOkay = 0;
156
uint8_t NCSerialDataOkay = 0;
154
uint8_t NCSerialDataOkay = 0;
157
 
155
 
158
uint8_t SPI_CommandSequence[] = { SPI_CMD_USER, SPI_CMD_STICK, SPI_CMD_PARAMETER1, SPI_CMD_STICK, SPI_CMD_MISC, SPI_CMD_VERSION };
156
uint8_t SPI_CommandSequence[] = { SPI_CMD_USER, SPI_CMD_STICK, SPI_CMD_PARAMETER1, SPI_CMD_STICK, SPI_CMD_MISC, SPI_CMD_VERSION };
159
uint8_t SPI_CommandCounter = 0;
157
uint8_t SPI_CommandCounter = 0;
160
 
158
 
161
/*********************************************/
159
/*********************************************/
162
/*  Initialize SPI interface to NaviCtrl     */
160
/*  Initialize SPI interface to NaviCtrl     */
163
/*********************************************/
161
/*********************************************/
164
void SPI_MasterInit(void) {
162
void SPI_MasterInit(void) {
165
  DDR_SPI |= (1<<DD_MOSI)|(1<<DD_SCK);    // Set MOSI and SCK output, all others input
163
  DDR_SPI |= (1<<DD_MOSI)|(1<<DD_SCK);    // Set MOSI and SCK output, all others input
166
  SLAVE_SELECT_DDR_PORT |= (1 << SPI_SLAVE_SELECT); // set Slave select port as output port
164
  SLAVE_SELECT_DDR_PORT |= (1 << SPI_SLAVE_SELECT); // set Slave select port as output port
167
 
165
 
168
  SPCR = (1<<SPE)|(1<<MSTR)|(1<<SPR1)|(0<<SPR0)|(0<<SPIE);   // Enable SPI, Master, set clock rate fck/64
166
  SPCR = (1<<SPE)|(1<<MSTR)|(1<<SPR1)|(0<<SPR0)|(0<<SPIE);   // Enable SPI, Master, set clock rate fck/64
169
  SPSR = 0;//(1<<SPI2X);
167
  SPSR = 0;//(1<<SPI2X);
170
 
168
 
171
  SLAVE_SELECT_PORT |=  (1 << SPI_SLAVE_SELECT); // Deselect Slave
169
  SLAVE_SELECT_PORT |=  (1 << SPI_SLAVE_SELECT); // Deselect Slave
172
 
170
 
173
  SPI_TxBuffer = (uint8_t *) &ToNaviCtrl; // set pointer to tx-buffer
171
  SPI_TxBuffer = (uint8_t *) &toNaviCtrl; // set pointer to tx-buffer
174
  SPITransferCompleted = 1;
172
  SPITransferCompleted = 1;
175
  // initialize data packet to NaviControl
173
  // initialize data packet to NaviControl
176
  ToNaviCtrl.Sync1 = SPI_TXSYNCBYTE1;
174
  toNaviCtrl.Sync1 = SPI_TXSYNCBYTE1;
177
  ToNaviCtrl.Sync2 = SPI_TXSYNCBYTE2;
175
  toNaviCtrl.Sync2 = SPI_TXSYNCBYTE2;
178
 
176
 
179
  ToNaviCtrl.Command = SPI_CMD_USER;
177
  toNaviCtrl.Command = SPI_CMD_USER;
180
  ToNaviCtrl.IntegralNick = 0;
178
  toNaviCtrl.IntegralPitch = 0;
181
  ToNaviCtrl.IntegralRoll = 0;
179
  toNaviCtrl.IntegralRoll = 0;
182
  NCSerialDataOkay = 0;
180
  NCSerialDataOkay = 0;
183
  NCDataOkay = 0;
181
  NCDataOkay = 0;
184
 
182
 
185
  SPI_RxDataValid = 0;
183
  SPI_RxDataValid = 0;
186
 
184
 
187
  SPI_VersionInfo.Major = VERSION_MAJOR;
185
  SPI_VersionInfo.Major = VERSION_MAJOR;
188
  SPI_VersionInfo.Minor = VERSION_MINOR;
186
  SPI_VersionInfo.Minor = VERSION_MINOR;
189
  SPI_VersionInfo.Patch = VERSION_PATCH;
187
  SPI_VersionInfo.Patch = VERSION_PATCH;
190
  SPI_VersionInfo.Compatible = NC_SPI_COMPATIBLE;
188
  SPI_VersionInfo.Compatible = NC_SPI_COMPATIBLE;
191
}
189
}
192
 
-
 
193
 
190
 
194
/**********************************************************/
191
/**********************************************************/
195
/*  Update Data transferd by the SPI from/to NaviCtrl     */
192
/*  Update Data transferd by the SPI from/to NaviCtrl     */
196
/**********************************************************/
193
/**********************************************************/
197
void UpdateSPI_Buffer(void) {
194
void UpdateSPI_Buffer(void) {
198
  uint8_t i;
195
  uint8_t i;
199
  int16_t tmp;
196
  int16_t tmp;
200
  cli(); // stop all interrupts to avoid writing of new data during update of that packet.
197
  cli(); // stop all interrupts to avoid writing of new data during update of that packet.
201
 
198
 
202
  // update content of packet to NaviCtrl
199
  // update content of packet to NaviCtrl
203
  ToNaviCtrl.IntegralNick = (int16_t)((10 * angle[PITCH]) / GYRO_DEG_FACTOR_PITCHROLL); // convert to multiple of 0.1°
200
  toNaviCtrl.IntegralPitch = (int16_t)((10 * angle[PITCH]) / GYRO_DEG_FACTOR_PITCHROLL); // convert to multiple of 0.1°
204
  ToNaviCtrl.IntegralRoll = (int16_t)((10 * angle[ROLL]) / GYRO_DEG_FACTOR_PITCHROLL); // convert to multiple of 0.1°
201
  toNaviCtrl.IntegralRoll  = (int16_t)((10 * angle[ROLL]) / GYRO_DEG_FACTOR_PITCHROLL);  // convert to multiple of 0.1°
205
  ToNaviCtrl.GyroHeading  = (int16_t)((10 * yawGyroHeading)   / GYRO_DEG_FACTOR_YAW); // convert to multiple of 0.1°
202
  toNaviCtrl.GyroHeading   = (int16_t)((10 * yawGyroHeading)   / GYRO_DEG_FACTOR_YAW);   // convert to multiple of 0.1°
206
  ToNaviCtrl.GyroNick = rate_PID[PITCH]; // TODO: Which one should it be??
203
  toNaviCtrl.GyroPitch = rate_ATT[PITCH];
207
  ToNaviCtrl.GyroRoll = rate_PID[ROLL];
204
  toNaviCtrl.GyroRoll = rate_ATT[ROLL];
208
  ToNaviCtrl.GyroYaw =  yawRate;
205
  toNaviCtrl.GyroYaw  = yawRate;
209
  ToNaviCtrl.AccNick =  0; // ((int16_t) 10 * ACC_AMPLIFY * (NaviAccNick / NaviCntAcc)) / ACC_DEG_FACTOR; // convert to multiple of 0.1°
206
  toNaviCtrl.AccPitch = (10 * getAngleEstimateFromAcc(PITCH)) / GYRO_DEG_FACTOR_PITCHROLL; // convert to multiple of 0.1°
210
  ToNaviCtrl.AccRoll =  0; // ((int16_t) 10 * ACC_AMPLIFY * (NaviAccRoll / NaviCntAcc)) / ACC_DEG_FACTOR; // convert to multiple of 0.1°
207
  toNaviCtrl.AccRoll  = (10 * getAngleEstimateFromAcc(ROLL))  / GYRO_DEG_FACTOR_PITCHROLL; // convert to multiple of 0.1°
211
  // naviCntAcc = 0; naviAccPitch = 0; naviAccRoll = 0;
-
 
-
 
208
 
-
 
209
 
-
 
210
  // TODO: What are these little bastards?
-
 
211
 
-
 
212
  averageAcc[PITCH] = averageAcc[ROLL] = averageAccCount = 0;
212
 
213
 
213
  switch(ToNaviCtrl.Command) {
214
  switch(toNaviCtrl.Command) {
214
  case SPI_CMD_USER:
215
  case SPI_CMD_USER:
215
    for (i=0; i<sizeof(dynamicParams.UserParams); i++) {
216
    for (i=0; i<sizeof(dynamicParams.UserParams); i++) {
216
      ToNaviCtrl.Param.Byte[i] = dynamicParams.UserParams[i];
217
      toNaviCtrl.Param.Byte[i] = dynamicParams.UserParams[i];
217
    }
218
    }
218
    ToNaviCtrl.Param.Byte[8] = MKFlags;
219
    toNaviCtrl.Param.Byte[8] = MKFlags;
219
    MKFlags &= ~(MKFLAG_CALIBRATE | MKFLAG_START); // calibrate and start are temporal states that are cleared immediately after transmitting
220
    MKFlags &= ~(MKFLAG_CALIBRATE | MKFLAG_START); // calibrate and start are temporal states that are cleared immediately after transmitting
220
    ToNaviCtrl.Param.Byte[9] = (uint8_t)UBat;
221
    toNaviCtrl.Param.Byte[9] = (uint8_t)UBat;
221
    ToNaviCtrl.Param.Byte[10] = staticParams.LowVoltageWarning;
222
    toNaviCtrl.Param.Byte[10] = staticParams.LowVoltageWarning;
222
    ToNaviCtrl.Param.Byte[11] = getActiveParamSet();
223
    toNaviCtrl.Param.Byte[11] = getActiveParamSet();
223
    break;
224
    break;
224
 
225
 
225
  case SPI_CMD_PARAMETER1:
226
  case SPI_CMD_PARAMETER1:
226
    ToNaviCtrl.Param.Byte[0] = staticParams.NaviGpsModeControl;     // Parameters for the Naviboard
227
    toNaviCtrl.Param.Byte[0] = staticParams.NaviGpsModeControl;     // Parameters for the Naviboard
227
    ToNaviCtrl.Param.Byte[1] = staticParams.NaviGpsGain;
228
    toNaviCtrl.Param.Byte[1] = staticParams.NaviGpsGain;
228
    ToNaviCtrl.Param.Byte[2] = staticParams.NaviGpsP;
229
    toNaviCtrl.Param.Byte[2] = staticParams.NaviGpsP;
229
    ToNaviCtrl.Param.Byte[3] = staticParams.NaviGpsI;
230
    toNaviCtrl.Param.Byte[3] = staticParams.NaviGpsI;
230
    ToNaviCtrl.Param.Byte[4] = staticParams.NaviGpsD;
231
    toNaviCtrl.Param.Byte[4] = staticParams.NaviGpsD;
231
    ToNaviCtrl.Param.Byte[5] = staticParams.NaviGpsACC;
232
    toNaviCtrl.Param.Byte[5] = staticParams.NaviGpsACC;
232
    ToNaviCtrl.Param.Byte[6] = staticParams.NaviGpsMinSat;
233
    toNaviCtrl.Param.Byte[6] = staticParams.NaviGpsMinSat;
233
    ToNaviCtrl.Param.Byte[7] = staticParams.NaviStickThreshold;
234
    toNaviCtrl.Param.Byte[7] = staticParams.NaviStickThreshold;
234
    ToNaviCtrl.Param.Byte[8] = staticParams.NaviOperatingRadius;
235
    toNaviCtrl.Param.Byte[8] = staticParams.NaviOperatingRadius;
235
    ToNaviCtrl.Param.Byte[9] = staticParams.NaviWindCorrection;
236
    toNaviCtrl.Param.Byte[9] = staticParams.NaviWindCorrection;
236
    ToNaviCtrl.Param.Byte[10] = staticParams.NaviSpeedCompensation;
237
    toNaviCtrl.Param.Byte[10] = staticParams.NaviSpeedCompensation;
237
    ToNaviCtrl.Param.Byte[11] = staticParams.NaviAngleLimitation;
238
    toNaviCtrl.Param.Byte[11] = staticParams.NaviAngleLimitation;
238
    break;
239
    break;
239
   
240
   
240
  case SPI_CMD_STICK:
241
  case SPI_CMD_STICK:
241
    tmp = PPM_in[staticParams.ChannelAssignment[CH_THROTTLE]];  if(tmp > 127) tmp = 127; else if(tmp < -128) tmp = -128;
242
    tmp = PPM_in[staticParams.ChannelAssignment[CH_THROTTLE]];  if(tmp > 127) tmp = 127; else if(tmp < -128) tmp = -128;
242
    ToNaviCtrl.Param.Byte[0] = (int8_t) tmp;
243
    toNaviCtrl.Param.Byte[0] = (int8_t) tmp;
243
    tmp = PPM_in[staticParams.ChannelAssignment[CH_YAW]]; if(tmp > 127) tmp = 127; else if(tmp < -128) tmp = -128;
244
    tmp = PPM_in[staticParams.ChannelAssignment[CH_YAW]]; if(tmp > 127) tmp = 127; else if(tmp < -128) tmp = -128;
244
    ToNaviCtrl.Param.Byte[1] = (int8_t) tmp;
245
    toNaviCtrl.Param.Byte[1] = (int8_t) tmp;
245
    tmp = PPM_in[staticParams.ChannelAssignment[CH_ROLL]]; if(tmp > 127) tmp = 127; else if(tmp < -128) tmp = -128;
246
    tmp = PPM_in[staticParams.ChannelAssignment[CH_ROLL]]; if(tmp > 127) tmp = 127; else if(tmp < -128) tmp = -128;
246
    ToNaviCtrl.Param.Byte[2] = (int8_t) tmp;
247
    toNaviCtrl.Param.Byte[2] = (int8_t) tmp;
247
    tmp = PPM_in[staticParams.ChannelAssignment[CH_PITCH]]; if(tmp > 127) tmp = 127; else if(tmp < -128) tmp = -128;
248
    tmp = PPM_in[staticParams.ChannelAssignment[CH_PITCH]]; if(tmp > 127) tmp = 127; else if(tmp < -128) tmp = -128;
248
    ToNaviCtrl.Param.Byte[3] = (int8_t) tmp;
249
    toNaviCtrl.Param.Byte[3] = (int8_t) tmp;
249
    ToNaviCtrl.Param.Byte[4] = (uint8_t) variables[0];
250
    toNaviCtrl.Param.Byte[4] = (uint8_t) variables[0];
250
    ToNaviCtrl.Param.Byte[5] = (uint8_t) variables[1];
251
    toNaviCtrl.Param.Byte[5] = (uint8_t) variables[1];
251
    ToNaviCtrl.Param.Byte[6] = (uint8_t) variables[2];
252
    toNaviCtrl.Param.Byte[6] = (uint8_t) variables[2];
252
    ToNaviCtrl.Param.Byte[7] = (uint8_t) variables[3];
253
    toNaviCtrl.Param.Byte[7] = (uint8_t) variables[3];
253
    ToNaviCtrl.Param.Byte[8] = (uint8_t) RC_Quality;
254
    toNaviCtrl.Param.Byte[8] = (uint8_t) RC_Quality;
254
    break;
255
    break;
255
 
256
 
256
  case SPI_CMD_MISC:
257
  case SPI_CMD_MISC:
257
    ToNaviCtrl.Param.Byte[0] = compassCalState;
258
    toNaviCtrl.Param.Byte[0] = compassCalState;
258
    if(compassCalState > 4)
259
    if(compassCalState > 4) { // jump from 5 to 0
259
      { // jump from 5 to 0
-
 
260
        compassCalState  = 0;
260
          compassCalState  = 0;
261
      }
261
    }
262
    ToNaviCtrl.Param.Byte[1] = staticParams.NaviPHLoginTime;
262
    toNaviCtrl.Param.Byte[1] = staticParams.NaviPHLoginTime;
-
 
263
    // TODO: Height and in the correct scaling...
263
    ToNaviCtrl.Param.Int[1]  = 0; //readingHeight; // at address of Byte 2 and 3
264
    toNaviCtrl.Param.Int[1]  = 0; //readingHeight; // at address of Byte 2 and 3
264
    ToNaviCtrl.Param.Byte[4] = staticParams.NaviGpsPLimit;
265
    toNaviCtrl.Param.Byte[4] = staticParams.NaviGpsPLimit;
265
    ToNaviCtrl.Param.Byte[5] = staticParams.NaviGpsILimit;
266
    toNaviCtrl.Param.Byte[5] = staticParams.NaviGpsILimit;
266
    ToNaviCtrl.Param.Byte[6] = staticParams.NaviGpsDLimit;
267
    toNaviCtrl.Param.Byte[6] = staticParams.NaviGpsDLimit;
267
    break;
268
    break;
268
   
269
   
269
  case SPI_CMD_VERSION:
270
  case SPI_CMD_VERSION:
270
    ToNaviCtrl.Param.Byte[0] = SPI_VersionInfo.Major;
271
    toNaviCtrl.Param.Byte[0] = SPI_VersionInfo.Major;
271
    ToNaviCtrl.Param.Byte[1] = SPI_VersionInfo.Minor;
272
    toNaviCtrl.Param.Byte[1] = SPI_VersionInfo.Minor;
272
    ToNaviCtrl.Param.Byte[2] = SPI_VersionInfo.Patch;
273
    toNaviCtrl.Param.Byte[2] = SPI_VersionInfo.Patch;
273
    ToNaviCtrl.Param.Byte[3] = SPI_VersionInfo.Compatible;
274
    toNaviCtrl.Param.Byte[3] = SPI_VersionInfo.Compatible;
274
    ToNaviCtrl.Param.Byte[4] = BoardRelease;
275
    toNaviCtrl.Param.Byte[4] = BoardRelease;
275
    break;
276
    break;
276
   
-
 
277
  default:
277
  default:
278
    break;
278
    break;
279
  }
279
  }
280
 
280
 
281
  sei(); // enable all interrupts
281
  sei(); // enable all interrupts
282
 
282
 
283
  // analyze content of packet from NaviCtrl if valid
283
  // analyze content of packet from NaviCtrl if valid
284
  if (SPI_RxDataValid) {
284
  if (SPI_RxDataValid) {
285
    // update gps controls
285
    // update gps controls
286
    if(abs(FromNaviCtrl.GPSStickNick) < 512 && abs(FromNaviCtrl.GPSStickRoll) < 512 && (staticParams.GlobalConfig & CFG_GPS_ACTIVE)) {
286
    if(abs(fromNaviCtrl.GPSStickPitch) < 512 && abs(fromNaviCtrl.GPSStickRoll) < 512 && (staticParams.GlobalConfig & CFG_GPS_ACTIVE)) {
287
      GPSStickPitch     = FromNaviCtrl.GPSStickNick;
287
      GPSStickPitch     = fromNaviCtrl.GPSStickPitch;
288
      GPSStickRoll      = FromNaviCtrl.GPSStickRoll;
288
      GPSStickRoll      = fromNaviCtrl.GPSStickRoll;
289
      NCDataOkay = 250;
289
      NCDataOkay = 250;
290
    }
290
    }
291
    // update compass readings
291
    // update compass readings
292
    if(FromNaviCtrl.CompassHeading <= 360) {
292
    if(fromNaviCtrl.CompassHeading <= 360) {
293
      compassHeading = FromNaviCtrl.CompassHeading;
293
      compassHeading = fromNaviCtrl.CompassHeading;
294
    }
294
    }
295
    if(compassHeading < 0) compassOffCourse = 0;
295
    if(compassHeading < 0) compassOffCourse = 0;
296
    else compassOffCourse = ((540 + compassHeading - compassCourse) % 360) - 180;
296
    else compassOffCourse = ((540 + compassHeading - compassCourse) % 360) - 180;
297
    // NaviCtrl wants to beep?
297
    // NaviCtrl wants to beep?
298
    if (FromNaviCtrl.BeepTime > BeepTime && !compassCalState) BeepTime = FromNaviCtrl.BeepTime;
298
    if (fromNaviCtrl.BeepTime > BeepTime && !compassCalState) BeepTime = fromNaviCtrl.BeepTime;
299
   
299
   
300
    switch (FromNaviCtrl.Command) {
300
    switch (fromNaviCtrl.Command) {
301
    case SPI_KALMAN:
301
    case SPI_KALMAN:
302
      dynamicParams.KalmanK = FromNaviCtrl.Param.Byte[0];
302
      dynamicParams.KalmanK             = fromNaviCtrl.Param.Byte[0];
303
      dynamicParams.KalmanMaxFusion = FromNaviCtrl.Param.Byte[1];
303
      dynamicParams.KalmanMaxFusion = fromNaviCtrl.Param.Byte[1];
304
      dynamicParams.KalmanMaxDrift = FromNaviCtrl.Param.Byte[2];
304
      dynamicParams.KalmanMaxDrift      = fromNaviCtrl.Param.Byte[2];
305
      NCSerialDataOkay = FromNaviCtrl.Param.Byte[3];
-
 
306
      //DebugOut.Analog[29] = NCSerialDataOkay;
305
      NCSerialDataOkay                          = fromNaviCtrl.Param.Byte[3];
307
      break;
306
      break;
308
     
307
     
309
    default:
308
    default:
310
      break;
309
      break;
311
    }
310
    }
312
  }
-
 
313
  else // no valid data from NaviCtrl
311
  } else { // no valid data from NaviCtrl
314
    {
-
 
315
      // disable GPS control
312
      // disable GPS control
316
      GPSStickPitch = 0;
313
      GPSStickPitch = 0;
317
      GPSStickRoll = 0;
314
      GPSStickRoll = 0;
318
    }
315
    }
319
}
316
}
320
 
317
 
321
/*********************************************/
318
/*********************************************/
322
/*  Start Transmission of packet to NaviCtrl */
319
/*  Start Transmission of packet to NaviCtrl */
323
/*********************************************/
320
/*********************************************/
324
void SPI_StartTransmitPacket(void){
321
void SPI_StartTransmitPacket(void){
325
  if (!SPITransferCompleted) return; // return immediately if transfer is in progress
322
  if (!SPITransferCompleted) return; // return immediately if transfer is in progress
326
  else // transmission was completed
323
  else // transmission was completed
327
    {
324
    {
328
      SLAVE_SELECT_PORT &=  ~(1 << SPI_SLAVE_SELECT);  // Select slave
325
      SLAVE_SELECT_PORT &=  ~(1 << SPI_SLAVE_SELECT);  // Select slave
329
     
326
     
330
      // cyclic commands
327
      // cyclic commands
331
      ToNaviCtrl.Command = SPI_CommandSequence[SPI_CommandCounter++];
328
      toNaviCtrl.Command = SPI_CommandSequence[SPI_CommandCounter++];
332
      if (SPI_CommandCounter >= sizeof(SPI_CommandSequence)) SPI_CommandCounter = 0;
329
      if (SPI_CommandCounter >= sizeof(SPI_CommandSequence)) SPI_CommandCounter = 0;
333
     
330
     
334
      SPITransferCompleted = 0; // transfer is in progress
331
      SPITransferCompleted = 0; // transfer is in progress
335
      UpdateSPI_Buffer();    // update data in ToNaviCtrl
332
      UpdateSPI_Buffer();    // update data in toNaviCtrl
336
     
333
     
337
      SPI_TxBufferIndex = 1; //proceed with 2nd byte
334
      SPI_TxBufferIndex = 1; //proceed with 2nd byte
338
     
335
     
339
      // -- Debug-Output ---
336
      // -- Debug-Output ---
340
      //----
337
      //----
341
      asm volatile ("nop"); asm volatile ("nop"); asm volatile ("nop"); asm volatile ("nop");      asm volatile ("nop"); asm volatile ("nop"); asm volatile ("nop"); asm volatile ("nop");
338
      asm volatile ("nop"); asm volatile ("nop"); asm volatile ("nop"); asm volatile ("nop");      asm volatile ("nop"); asm volatile ("nop"); asm volatile ("nop"); asm volatile ("nop");
342
      asm volatile ("nop"); asm volatile ("nop"); asm volatile ("nop"); asm volatile ("nop");      asm volatile ("nop"); asm volatile ("nop"); asm volatile ("nop"); asm volatile ("nop");
339
      asm volatile ("nop"); asm volatile ("nop"); asm volatile ("nop"); asm volatile ("nop");      asm volatile ("nop"); asm volatile ("nop"); asm volatile ("nop"); asm volatile ("nop");
343
      asm volatile ("nop"); asm volatile ("nop"); asm volatile ("nop"); asm volatile ("nop");      asm volatile ("nop"); asm volatile ("nop"); asm volatile ("nop"); asm volatile ("nop");
340
      asm volatile ("nop"); asm volatile ("nop"); asm volatile ("nop"); asm volatile ("nop");      asm volatile ("nop"); asm volatile ("nop"); asm volatile ("nop"); asm volatile ("nop");
344
      ToNaviCtrl.Chksum = ToNaviCtrl.Sync1; // init checksum
341
      toNaviCtrl.Chksum = toNaviCtrl.Sync1; // init checksum
345
      SPDR = ToNaviCtrl.Sync1; // send first byte
342
      SPDR = toNaviCtrl.Sync1; // send first byte
346
    }
343
    }
347
}
344
}
348
 
345
 
349
//------------------------------------------------------
346
//------------------------------------------------------
350
// This is the spi data transfer between FlightCtrl and NaviCtrl
347
// This is the spi data transfer between FlightCtrl and NaviCtrl
351
// Every time this routine is called within the mainloop one byte of the packet to
348
// Every time this routine is called within the mainloop one byte of the packet to
352
// the NaviCtrl and one byte of the packet from the NaviCtrl is possible transfered
349
// the NaviCtrl and one byte of the packet from the NaviCtrl is possible transfered
353
 
350
 
354
void SPI_TransmitByte(void) {
351
void SPI_TransmitByte(void) {
355
  static SPI_RXState_t SPI_RXState = SPI_SYNC1;
352
  static SPI_RXState_t SPI_RXState = SPI_SYNC1;
356
  uint8_t rxdata;
353
  uint8_t rxdata;
357
  static uint8_t rxchksum;
354
  static uint8_t rxchksum;
358
 
355
 
359
  if (SPITransferCompleted) return;  // return immediatly if transfer was completed
356
  if (SPITransferCompleted) return;  // return immediatly if transfer was completed
360
  if (!(SPSR & (1 << SPIF))) return; // return if no SPI-IRQ pending
357
  if (!(SPSR & (1 << SPIF))) return; // return if no SPI-IRQ pending
361
  SendSPI = 4; // mait 4 * 0.102 ms for the next call of SPI_TransmitByte() in the main loop
358
  SendSPI = 4; // mait 4 * 0.102 ms for the next call of SPI_TransmitByte() in the main loop
362
 
359
 
363
  SLAVE_SELECT_PORT |=  (1 << SPI_SLAVE_SELECT);   // DeselectSlave
360
  SLAVE_SELECT_PORT |=  (1 << SPI_SLAVE_SELECT);   // DeselectSlave
364
 
361
 
365
  rxdata = SPDR; // save spi data register
362
  rxdata = SPDR; // save spi data register
366
 
363
 
367
  switch (SPI_RXState) {
364
  switch (SPI_RXState) {
368
  case SPI_SYNC1: // first sync byte
365
  case SPI_SYNC1: // first sync byte
369
    SPI_RxBufferIndex = 0; // set pointer to start of rx buffer
366
    SPI_RxBufferIndex = 0; // set pointer to start of rx buffer
370
    rxchksum = rxdata; // initialize checksum
367
    rxchksum = rxdata; // initialize checksum
371
    if (rxdata == SPI_RXSYNCBYTE1 )
368
    if (rxdata == SPI_RXSYNCBYTE1 )
372
      { // 1st Syncbyte found
369
      { // 1st Syncbyte found
373
        SPI_RXState  = SPI_SYNC2; // trigger to state for second sync byte
370
        SPI_RXState  = SPI_SYNC2; // trigger to state for second sync byte
374
      }
371
      }
375
    break;
372
    break;
376
   
373
   
377
  case SPI_SYNC2: // second sync byte
374
  case SPI_SYNC2: // second sync byte
378
    if (rxdata == SPI_RXSYNCBYTE2)
375
    if (rxdata == SPI_RXSYNCBYTE2)
379
      { // 2nd Syncbyte found
376
      { // 2nd Syncbyte found
380
        rxchksum += rxdata; // update checksum
377
        rxchksum += rxdata; // update checksum
381
        SPI_RXState  = SPI_DATA;   // trigger to state for second sync byte
378
        SPI_RXState  = SPI_DATA;   // trigger to state for second sync byte
382
      }
379
      }
383
    else // 2nd Syncbyte not found
380
    else // 2nd Syncbyte not found
384
      {
381
      {
385
        SPI_RXState  = SPI_SYNC1; // jump back to 1st sync byte
382
        SPI_RXState  = SPI_SYNC1; // jump back to 1st sync byte
386
      }
383
      }
387
    break;
384
    break;
388
   
385
   
389
  case SPI_DATA: // data bytes
386
  case SPI_DATA: // data bytes
390
    SPI_RxBuffer[SPI_RxBufferIndex++] = rxdata;  // copy data byte to spi buffer
387
    SPI_RxBuffer[SPI_RxBufferIndex++] = rxdata;  // copy data byte to spi buffer
391
    // if all bytes are received of a packet from the NaviCtrl
388
    // if all bytes are received of a packet from the NaviCtrl
392
    if (SPI_RxBufferIndex >= SPI_RXBUFFER_LEN)
-
 
393
      {   // last byte transfered is the checksum of the packet
389
    if (SPI_RxBufferIndex >= SPI_RXBUFFER_LEN) {   // last byte transfered is the checksum of the packet
394
        if (rxdata == rxchksum) // checksum matching?
390
        if (rxdata == rxchksum) { // checksum matching?
395
          {
-
 
396
            // copy SPI_RxBuffer -> FromFlightCtrl
391
            // copy SPI_RxBuffer -> FromFlightCtrl
397
            uint8_t *ptr = (uint8_t *)&FromNaviCtrl;
392
            uint8_t *ptr = (uint8_t *)&fromNaviCtrl;
398
            cli();
393
            cli();
399
            memcpy(ptr, (uint8_t *) SPI_RxBuffer, sizeof(FromNaviCtrl));
394
            memcpy(ptr, (uint8_t *) SPI_RxBuffer, sizeof(fromNaviCtrl));
400
            sei();
395
            sei();
401
            SPI_RxDataValid = 1;
396
            SPI_RxDataValid = 1;
402
            //DebugOut.Analog[18]++;
-
 
403
          }
-
 
404
        else
-
 
405
          {   // checksum does not match
397
          } else {   // checksum does not match
406
            //DebugOut.Analog[17]++;
-
 
407
            SPI_RxDataValid = 0; // reset valid flag
398
            SPI_RxDataValid = 0; // reset valid flag
408
          }
399
          }
409
        SPI_RXState  = SPI_SYNC1; // reset state sync
400
          SPI_RXState  = SPI_SYNC1; // reset state sync
410
      }
-
 
411
    else // not all bytes transfered
401
    } else { // not all bytes transfered
412
      {
-
 
413
        rxchksum += rxdata; // update checksum
402
          rxchksum += rxdata; // update checksum
414
      }
403
    }
415
    break;
404
    break;
416
  }// eof switch(SPI_RXState)
405
  }// eof switch(SPI_RXState)
417
 
406
 
418
  // if still some bytes left for transmission to NaviCtrl
407
  // if still some bytes left for transmission to NaviCtrl
419
  if (SPI_TxBufferIndex < SPI_TXBUFFER_LEN)
408
  if (SPI_TxBufferIndex < SPI_TXBUFFER_LEN) {
420
    {
-
 
421
      SLAVE_SELECT_PORT &=  ~(1 << SPI_SLAVE_SELECT);  // SelectSlave
409
      SLAVE_SELECT_PORT &=  ~(1 << SPI_SLAVE_SELECT);  // SelectSlave
422
      asm volatile ("nop"); asm volatile ("nop"); asm volatile ("nop"); asm volatile ("nop"); asm volatile ("nop"); asm volatile ("nop"); asm volatile ("nop"); asm volatile ("nop");
410
      asm volatile ("nop"); asm volatile ("nop"); asm volatile ("nop"); asm volatile ("nop"); asm volatile ("nop"); asm volatile ("nop"); asm volatile ("nop"); asm volatile ("nop");
423
      asm volatile ("nop"); asm volatile ("nop"); asm volatile ("nop"); asm volatile ("nop"); asm volatile ("nop"); asm volatile ("nop"); asm volatile ("nop"); asm volatile ("nop");
411
      asm volatile ("nop"); asm volatile ("nop"); asm volatile ("nop"); asm volatile ("nop"); asm volatile ("nop"); asm volatile ("nop"); asm volatile ("nop"); asm volatile ("nop");
424
      asm volatile ("nop"); asm volatile ("nop"); asm volatile ("nop"); asm volatile ("nop");   asm volatile ("nop"); asm volatile ("nop"); asm volatile ("nop"); asm volatile ("nop");
412
      asm volatile ("nop"); asm volatile ("nop"); asm volatile ("nop"); asm volatile ("nop");   asm volatile ("nop"); asm volatile ("nop"); asm volatile ("nop"); asm volatile ("nop");
425
     
413
     
426
      SPDR = SPI_TxBuffer[SPI_TxBufferIndex]; // transmit byte
414
      SPDR = SPI_TxBuffer[SPI_TxBufferIndex]; // transmit byte
427
      ToNaviCtrl.Chksum += SPI_TxBuffer[SPI_TxBufferIndex]; // update checksum for everey byte that was sent
415
      toNaviCtrl.Chksum += SPI_TxBuffer[SPI_TxBufferIndex]; // update checksum for everey byte that was sent
428
      SPI_TxBufferIndex++;
416
      SPI_TxBufferIndex++;
429
    } else {
417
    } else {
430
      //Transfer of all bytes of the packet to NaviCtrl completed
418
      //Transfer of all bytes of the packet to NaviCtrl completed
431
      SPITransferCompleted = 1;
419
      SPITransferCompleted = 1;
432
    }
420
    }
433
}
421
}
434
 
422