Subversion Repositories FlightCtrl

Rev

Details | Last modification | View Log | RSS feed

Rev Author Line No. Line
2751 - 1
/*############################################################################
2
############################################################################*/
3
 
4
#include "main.h"
5
 
6
volatile unsigned char twi_state = 0;
7
unsigned char motor = 0;
8
unsigned char motorread = 0,MissingMotor = 0;
9
 
10
MotorData_t Motor[MAX_MOTORS];
11
 
12
unsigned int I2CError = 0;
13
 
14
//############################################################################
15
//Initzialisieren der I2C (TWI) Schnittstelle
16
void i2c_init(void)
17
//############################################################################
18
{
19
  TWSR = 0;
20
  TWBR = ((SYSCLK/SCL_CLOCK)-16)/2;
21
}
22
 
23
//############################################################################
24
//Start I2C
25
void i2c_start(void)
26
//############################################################################
27
{
28
    TWCR = (1<<TWSTA) | (1<<TWEN) | (1<<TWINT) | (1<<TWIE);
29
}
30
 
31
//############################################################################
32
void i2c_stop(void)
33
//############################################################################
34
{
35
    TWCR = (1<<TWEN) | (1<<TWSTO) | (1<<TWINT);
36
}
37
 
38
void i2c_reset(void)
39
//############################################################################
40
{
41
                 i2c_stop();
42
                 twi_state = 0;
43
                 motor = TWDR;
44
                 motor = 0;
45
                 TWCR = 0x80;
46
                 TWAMR = 0;
47
                 TWAR = 0;
48
                 TWDR = 0;
49
                 TWSR = 0;
50
                 TWBR = 0;
51
                 i2c_init();
52
                 i2c_start();
53
                 i2c_write_byte(0);
54
}
55
 
56
//############################################################################
57
void i2c_write_byte(char byte)
58
//############################################################################
59
{
60
    TWSR = 0x00;
61
    TWDR = byte;
62
    TWCR = (1<<TWINT) | (1<<TWEN) | (1<<TWIE);
63
}
64
 
65
/****************************************/
66
/*    Write to I2C                      */
67
/****************************************/
68
void I2C_WriteByte(int8_t byte)
69
{
70
    // move byte to send into TWI Data Register
71
    TWDR = byte;
72
    // clear interrupt flag (TWINT = 1)
73
    // enable i2c bus (TWEN = 1)
74
    // enable interrupt (TWIE = 1)
75
    TWCR = (1<<TWINT) | (1<<TWEN) | (1<<TWIE);
76
}
77
 
78
/****************************************/
79
/*    Receive byte and send ACK         */
80
/****************************************/
81
void I2C_ReceiveByte(void)
82
{
83
   TWCR = (1<<TWINT) | (1<<TWEN) | (1<<TWIE) | (1<<TWEA);
84
}
85
 
86
/****************************************/
87
/* I2C receive last byte and send no ACK*/
88
/****************************************/
89
void I2C_ReceiveLastByte(void)
90
{
91
   TWCR = (1<<TWINT) | (1<<TWEN) | (1<<TWIE);
92
}
93
 
94
 
95
 
96
//############################################################################
97
SIGNAL (TWI_vect)
98
//############################################################################
99
{
100
 static unsigned char missing_motor;
101
     switch(twi_state++)
102
        {
103
//++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
104
// Writing the Data
105
//++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
106
        case 0:
107
                        while(Mixer.Motor[motor][0] <= 0 && motor < MAX_MOTORS) motor++;  // skip if not used
108
                                if(motor == MAX_MOTORS)  // writing finished -> now read
109
                                {
110
                                 motor = 0;
111
                                 twi_state = 3;
112
                                 i2c_write_byte(0x53+(motorread*2));
113
                                 }
114
                                else i2c_write_byte(0x52+(motor*2));
115
                break;
116
        case 1:
117
                i2c_write_byte(Motor[motor++].SetPoint);
118
                break;
119
        case 2:
120
                if(TWSR == 0x30)
121
                                 {
122
                                  if(!missing_motor) missing_motor = motor;
123
                                  if((Motor[motor-1].State & MOTOR_STATE_ERROR_MASK) < MOTOR_STATE_ERROR_MASK) Motor[motor-1].State++; // increment error counter and handle overflow
124
                                 }
125
                i2c_stop();
126
                I2CTimeout = 10;
127
                twi_state = 0;
128
                i2c_start();
129
                break;
130
//++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
131
// Reading Data
132
//++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
133
        case 3:
134
               //Transmit 1st byte for reading
135
                if(TWSR != 0x40)  // Error?
136
                 {
137
                  Motor[motorread].State &= ~MOTOR_STATE_PRESENT_MASK; // clear present bit
138
                  motorread++;
139
                  if(motorread >= MAX_MOTORS) motorread = 0;
140
                  i2c_stop();
141
                  twi_state = 0;
142
                 }
143
                 else
144
                 {
145
                  Motor[motorread].State |= MOTOR_STATE_PRESENT_MASK; // set present bit
146
                  I2C_ReceiveByte();
147
                 }
148
                MissingMotor = missing_motor;
149
                missing_motor = 0;
150
                break;
151
        case 4: //Read 1st byte and transmit 2nd Byte
152
                Motor[motorread].Current = TWDR;
153
                I2C_ReceiveLastByte(); //nack
154
                break;
155
        case 5:
156
                //Read 2nd byte
157
                Motor[motorread].MaxPWM = TWDR;
158
                                motorread++; // next motor
159
                if(motorread >= MAX_MOTORS) motorread = 0;
160
                i2c_stop();
161
                twi_state = 0;
162
                break;
163
 
164
//++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
165
// writing Gyro-Offset
166
//++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
167
        case 8:
168
                i2c_write_byte(0x98); // Address of the DAC
169
                break;
170
        case 9:
171
                i2c_write_byte(0x10); // Update Channel A
172
                break;
173
        case 10:
174
                i2c_write_byte(AnalogOffsetNick); // Value
175
                break;
176
        case 11:
177
                i2c_write_byte(0x80); // Value
178
                break;
179
        case 12:
180
                i2c_stop();
181
                I2CTimeout = 10;
182
                i2c_start();
183
                break;
184
        case 13:
185
                i2c_write_byte(0x98); // Address of the DAC
186
                break;
187
        case 14:
188
                i2c_write_byte(0x12); // Update Channel B
189
                break;
190
        case 15:
191
                i2c_write_byte(AnalogOffsetRoll); // Value
192
                break;
193
        case 16:
194
                i2c_write_byte(0x80); // Value
195
                break;
196
        case 17:
197
                i2c_stop();
198
                I2CTimeout = 10;
199
                i2c_start();
200
                break;
201
        case 18:
202
                i2c_write_byte(0x98); // Address of the DAC
203
                break;
204
        case 19:
205
                i2c_write_byte(0x14); // Update Channel C
206
                break;
207
        case 20:
208
                i2c_write_byte(AnalogOffsetGier); // Value
209
                break;
210
        case 21:
211
                i2c_write_byte(0x80); // Value
212
                break;
213
        case 22:
214
                i2c_stop();
215
                I2CTimeout = 10;
216
                twi_state = 0;
217
                break;
218
        default: twi_state = 0;
219
                break;
220
                }
221
 TWCR |= 0x80;
222
}