3 Copyright (C) Dean Camera, 2009.
5 dean [at] fourwalledcubicle [dot] com
6 www.fourwalledcubicle.com
10 Copyright 2009 Dean Camera (dean [at] fourwalledcubicle [dot] com)
12 Permission to use, copy, modify, and distribute this software
13 and its documentation for any purpose and without fee is hereby
14 granted, provided that the above copyright notice appear in all
15 copies and that both that the copyright notice and this
16 permission notice and warranty disclaimer appear in supporting
17 documentation, and that the name of the author not be used in
18 advertising or publicity pertaining to distribution of the
19 software without specific, written prior permission.
21 The author disclaim all warranties with regard to this
22 software, including all implied warranties of merchantability
23 and fitness. In no event shall the author be liable for any
24 special, indirect or consequential damages or any damages
25 whatsoever resulting from loss of use, data or profits, whether
26 in an action of contract, negligence or other tortious action,
27 arising out of or in connection with the use or performance of
33 * V2Protocol handler, to process V2 Protocol commands used in Atmel programmer devices.
36 #define INCLUDE_FROM_V2PROTOCOL_C
37 #include "V2Protocol.h"
39 void V2Protocol_ProcessCommand(void)
41 uint8_t V2Command
= Endpoint_Read_Byte();
46 V2Protocol_Command_SignOn();
48 case CMD_SET_PARAMETER
:
49 case CMD_GET_PARAMETER
:
50 V2Protocol_Command_GetSetParam(V2Command
);
52 case CMD_LOAD_ADDRESS
:
53 V2Protocol_Command_LoadAddress();
55 case CMD_ENTER_PROGMODE_ISP
:
56 V2Protocol_Command_EnterISPMode();
58 case CMD_LEAVE_PROGMODE_ISP
:
59 V2Protocol_Command_LeaveISPMode();
61 case CMD_CHIP_ERASE_ISP
:
62 V2Protocol_Command_ChipErase();
64 case CMD_READ_FUSE_ISP
:
65 case CMD_READ_LOCK_ISP
:
66 case CMD_READ_SIGNATURE_ISP
:
67 case CMD_READ_OSCCAL_ISP
:
68 V2Protocol_Command_ReadFuseLockSigOSCCAL(V2Command
);
70 case CMD_PROGRAM_FUSE_ISP
:
71 case CMD_PROGRAM_LOCK_ISP
:
72 V2Protocol_Command_WriteFuseLock(V2Command
);
75 V2Protocol_Command_SPIMulti();
78 V2Protocol_Command_Unknown(V2Command
);
82 Endpoint_WaitUntilReady();
83 Endpoint_SetEndpointDirection(ENDPOINT_DIR_OUT
);
86 static void V2Protocol_Command_Unknown(uint8_t V2Command
)
88 /* Discard all incomming data */
89 while (Endpoint_BytesInEndpoint() == AVRISP_DATA_EPSIZE
)
92 Endpoint_WaitUntilReady();
96 Endpoint_SetEndpointDirection(ENDPOINT_DIR_IN
);
98 Endpoint_Write_Byte(V2Command
);
99 Endpoint_Write_Byte(STATUS_CMD_UNKNOWN
);
103 static void V2Protocol_Command_SignOn(void)
106 Endpoint_SetEndpointDirection(ENDPOINT_DIR_IN
);
108 Endpoint_Write_Byte(CMD_SIGN_ON
);
109 Endpoint_Write_Byte(STATUS_CMD_OK
);
110 Endpoint_Write_Byte(sizeof(PROGRAMMER_ID
) - 1);
111 Endpoint_Write_Stream_LE(PROGRAMMER_ID
, (sizeof(PROGRAMMER_ID
) - 1));
115 static void V2Protocol_Command_GetSetParam(uint8_t V2Command
)
117 uint8_t ParamID
= Endpoint_Read_Byte();
120 if (V2Command
== CMD_SET_PARAMETER
)
121 ParamValue
= Endpoint_Read_Byte();
124 Endpoint_SetEndpointDirection(ENDPOINT_DIR_IN
);
126 Endpoint_Write_Byte(V2Command
);
128 uint8_t ParamPrivs
= V2Params_GetParameterPrivellages(ParamID
);
130 if ((V2Command
== CMD_SET_PARAMETER
) && (ParamPrivs
& PARAM_PRIV_WRITE
))
132 Endpoint_Write_Byte(STATUS_CMD_OK
);
133 V2Params_SetParameterValue(ParamID
, ParamValue
);
135 else if ((V2Command
== CMD_GET_PARAMETER
) && (ParamPrivs
& PARAM_PRIV_READ
))
137 Endpoint_Write_Byte(STATUS_CMD_OK
);
138 Endpoint_Write_Byte(V2Params_GetParameterValue(ParamID
));
142 Endpoint_Write_Byte(STATUS_CMD_FAILED
);
148 static void V2Protocol_Command_LoadAddress(void)
150 Endpoint_Read_Stream_LE(&CurrentAddress
, sizeof(CurrentAddress
));
153 Endpoint_SetEndpointDirection(ENDPOINT_DIR_IN
);
155 if (CurrentAddress
& (1UL << 31))
156 V2Protocol_LoadExtendedAddress();
158 Endpoint_Write_Byte(CMD_LOAD_ADDRESS
);
159 Endpoint_Write_Byte(STATUS_CMD_OK
);
163 static void V2Protocol_Command_EnterISPMode(void)
168 uint8_t PinStabDelayMS
;
169 uint8_t ExecutionDelayMS
;
174 uint8_t EnterProgBytes
[4];
177 Endpoint_Read_Stream_LE(&Enter_ISP_Params
, sizeof(Enter_ISP_Params
));
180 Endpoint_SetEndpointDirection(ENDPOINT_DIR_IN
);
182 uint8_t ResponseStatus
= STATUS_CMD_FAILED
;
186 V2Protocol_DelayMS(Enter_ISP_Params
.ExecutionDelayMS
);
187 SPI_Init(V2Protocol_GetSPIPrescalerMask() | SPI_SCK_LEAD_RISING
| SPI_SAMPLE_LEADING
| SPI_MODE_MASTER
);
188 V2Protocol_ChangeTargetResetLine(true);
189 V2Protocol_DelayMS(Enter_ISP_Params
.PinStabDelayMS
);
191 while (Enter_ISP_Params
.SynchLoops
-- && (ResponseStatus
== STATUS_CMD_FAILED
))
193 uint8_t ResponseBytes
[4];
195 for (uint8_t RByte
= 0; RByte
< sizeof(ResponseBytes
); RByte
++)
197 V2Protocol_DelayMS(Enter_ISP_Params
.ByteDelay
);
198 ResponseBytes
[RByte
] = SPI_TransferByte(Enter_ISP_Params
.EnterProgBytes
[RByte
]);
201 /* Check if polling disabled, or if the polled value matches the expected value */
202 if (!Enter_ISP_Params
.PollIndex
|| (ResponseBytes
[Enter_ISP_Params
.PollIndex
- 1] == Enter_ISP_Params
.PollValue
))
203 ResponseStatus
= STATUS_CMD_OK
;
206 Endpoint_Write_Byte(CMD_ENTER_PROGMODE_ISP
);
207 Endpoint_Write_Byte(ResponseStatus
);
211 static void V2Protocol_Command_LeaveISPMode(void)
219 Endpoint_Read_Stream_LE(&Leave_ISP_Params
, sizeof(Leave_ISP_Params
));
222 Endpoint_SetEndpointDirection(ENDPOINT_DIR_IN
);
224 V2Protocol_DelayMS(Leave_ISP_Params
.PreDelayMS
);
225 V2Protocol_ChangeTargetResetLine(false);
227 V2Protocol_DelayMS(Leave_ISP_Params
.PostDelayMS
);
229 Endpoint_Write_Byte(CMD_LEAVE_PROGMODE_ISP
);
230 Endpoint_Write_Byte(STATUS_CMD_OK
);
234 static void V2Protocol_Command_ChipErase(void)
238 uint8_t EraseDelayMS
;
240 uint8_t EraseCommandBytes
[4];
243 Endpoint_Read_Stream_LE(&Erase_Chip_Params
, sizeof(Erase_Chip_Params
));
246 Endpoint_SetEndpointDirection(ENDPOINT_DIR_IN
);
248 uint8_t ResponseStatus
= STATUS_CMD_OK
;
250 for (uint8_t SByte
= 0; SByte
< sizeof(Erase_Chip_Params
.EraseCommandBytes
); SByte
++)
251 SPI_SendByte(Erase_Chip_Params
.EraseCommandBytes
[SByte
]);
253 if (Erase_Chip_Params
.PollMethod
== 0)
254 V2Protocol_DelayMS(Erase_Chip_Params
.EraseDelayMS
);
256 ResponseStatus
= V2Protocol_WaitWhileTargetBusy();
258 Endpoint_Write_Byte(CMD_CHIP_ERASE_ISP
);
259 Endpoint_Write_Byte(ResponseStatus
);
263 static void V2Protocol_Command_ReadFuseLockSigOSCCAL(uint8_t V2Command
)
268 uint8_t ReadCommandBytes
[4];
269 } Read_FuseLockSigOSCCAL_Params
;
271 Endpoint_Read_Stream_LE(&Read_FuseLockSigOSCCAL_Params
, sizeof(Read_FuseLockSigOSCCAL_Params
));
274 Endpoint_SetEndpointDirection(ENDPOINT_DIR_IN
);
276 uint8_t ResponseBytes
[4];
278 for (uint8_t RByte
= 0; RByte
< sizeof(ResponseBytes
); RByte
++)
279 ResponseBytes
[RByte
] = SPI_TransferByte(Read_FuseLockSigOSCCAL_Params
.ReadCommandBytes
[RByte
]);
281 Endpoint_Write_Byte(V2Command
);
282 Endpoint_Write_Byte(STATUS_CMD_OK
);
283 Endpoint_Write_Byte(ResponseBytes
[Read_FuseLockSigOSCCAL_Params
.RetByte
]);
284 Endpoint_Write_Byte(STATUS_CMD_OK
);
288 static void V2Protocol_Command_WriteFuseLock(uint8_t V2Command
)
292 uint8_t WriteCommandBytes
[4];
293 } Write_FuseLockSig_Params
;
295 Endpoint_Read_Stream_LE(&Write_FuseLockSig_Params
, sizeof(Write_FuseLockSig_Params
));
298 Endpoint_SetEndpointDirection(ENDPOINT_DIR_IN
);
300 for (uint8_t SByte
= 0; SByte
< sizeof(Write_FuseLockSig_Params
.WriteCommandBytes
); SByte
++)
301 SPI_SendByte(Write_FuseLockSig_Params
.WriteCommandBytes
[SByte
]);
303 Endpoint_Write_Byte(V2Command
);
304 Endpoint_Write_Byte(STATUS_CMD_OK
);
305 Endpoint_Write_Byte(STATUS_CMD_OK
);
309 static void V2Protocol_Command_SPIMulti(void)
319 Endpoint_Read_Stream_LE(&SPI_Multi_Params
, sizeof(SPI_Multi_Params
) -
320 sizeof(SPI_Multi_Params
.TxData
));
321 Endpoint_Read_Stream_LE(&SPI_Multi_Params
.TxData
, SPI_Multi_Params
.TxBytes
);
324 Endpoint_SetEndpointDirection(ENDPOINT_DIR_IN
);
326 Endpoint_Write_Byte(CMD_SPI_MULTI
);
327 Endpoint_Write_Byte(STATUS_CMD_OK
);
329 uint8_t CurrTxPos
= 0;
330 uint8_t CurrRxPos
= 0;
332 /* Write out bytes to transmit until the start of the bytes to receive is met */
333 while (CurrTxPos
< SPI_Multi_Params
.RxStartAddr
)
335 if (CurrTxPos
< SPI_Multi_Params
.TxBytes
)
336 SPI_SendByte(SPI_Multi_Params
.TxData
[CurrTxPos
]);
343 /* Transmit remaining bytes with padding as needed, read in response bytes */
344 while (CurrRxPos
< SPI_Multi_Params
.RxBytes
)
346 if (CurrTxPos
< SPI_Multi_Params
.TxBytes
)
347 Endpoint_Write_Byte(SPI_TransferByte(SPI_Multi_Params
.TxData
[CurrTxPos
++]));
349 Endpoint_Write_Byte(SPI_ReceiveByte());
354 Endpoint_Write_Byte(STATUS_CMD_OK
);