Blame view

lufa/USB_gadget.c 8.19 KB
41f6cbfd   dmohamed   first commit
1
2
  /*
               LUFA Library
1f4f9121   dmohamed   UPDATE:Inspiratio...
3
       Copyright (C) Dean Camera, 2017.
41f6cbfd   dmohamed   first commit
4
5
  
    dean [at] fourwalledcubicle [dot] com
1f4f9121   dmohamed   UPDATE:Inspiratio...
6
             www.lufa-lib.org
41f6cbfd   dmohamed   first commit
7
8
9
  */
  
  /*
1f4f9121   dmohamed   UPDATE:Inspiratio...
10
    Copyright 2017  Dean Camera (dean [at] fourwalledcubicle [dot] com)
41f6cbfd   dmohamed   first commit
11
12
13
14
15
16
17
18
19
20
  
    Permission to use, copy, modify, distribute, and sell this
    software and its documentation for any purpose is hereby granted
    without fee, provided that the above copyright notice appear in
    all copies and that both that the copyright notice and this
    permission notice and warranty disclaimer appear in supporting
    documentation, and that the name of the author not be used in
    advertising or publicity pertaining to distribution of the
    software without specific, written prior permission.
  
1f4f9121   dmohamed   UPDATE:Inspiratio...
21
    The author disclaims all warranties with regard to this
41f6cbfd   dmohamed   first commit
22
23
24
25
26
27
28
29
30
31
32
    software, including all implied warranties of merchantability
    and fitness.  In no event shall the author be liable for any
    special, indirect or consequential damages or any damages
    whatsoever resulting from loss of use, data or profits, whether
    in an action of contract, negligence or other tortious action,
    arising out of or in connection with the use or performance of
    this software.
  */
  
  /** \file
   *
57f5d0bc   dmohamed   FIX - Oublie de m...
33
   *  Main source file for the USBtoSerial project. This file contains the main tasks of
1f4f9121   dmohamed   UPDATE:Inspiratio...
34
   *  the project and is responsible for the initial application hardware configuration.
41f6cbfd   dmohamed   first commit
35
36
   */
  
1f4f9121   dmohamed   UPDATE:Inspiratio...
37
  #include "USB_gadget.h"
57f5d0bc   dmohamed   FIX - Oublie de m...
38
39
  
  /** Circular buffer to hold data from the host before it is sent to the device via the serial port. */
1f4f9121   dmohamed   UPDATE:Inspiratio...
40
41
42
43
  static RingBuffer_t USBtoUSART_Buffer;
  
  /** Underlying data buffer for \ref USBtoUSART_Buffer, where the stored bytes are located. */
  static uint8_t      USBtoUSART_Buffer_Data[128];
57f5d0bc   dmohamed   FIX - Oublie de m...
44
45
  
  /** Circular buffer to hold data from the serial port before it is sent to the host. */
1f4f9121   dmohamed   UPDATE:Inspiratio...
46
47
48
49
  static RingBuffer_t USARTtoUSB_Buffer;
  
  /** Underlying data buffer for \ref USARTtoUSB_Buffer, where the stored bytes are located. */
  static uint8_t      USARTtoUSB_Buffer_Data[128];
41f6cbfd   dmohamed   first commit
50
51
52
53
54
55
56
57
58
  
  /** LUFA CDC Class driver interface configuration and state information. This structure is
   *  passed to all CDC Class driver functions, so that multiple instances of the same class
   *  within a device can be differentiated from one another.
   */
  USB_ClassInfo_CDC_Device_t VirtualSerial_CDC_Interface =
  	{
  		.Config =
  			{
1f4f9121   dmohamed   UPDATE:Inspiratio...
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
  				.ControlInterfaceNumber         = INTERFACE_ID_CDC_CCI,
  				.DataINEndpoint                 =
  					{
  						.Address                = CDC_TX_EPADDR,
  						.Size                   = CDC_TXRX_EPSIZE,
  						.Banks                  = 1,
  					},
  				.DataOUTEndpoint                =
  					{
  						.Address                = CDC_RX_EPADDR,
  						.Size                   = CDC_TXRX_EPSIZE,
  						.Banks                  = 1,
  					},
  				.NotificationEndpoint           =
  					{
  						.Address                = CDC_NOTIFICATION_EPADDR,
  						.Size                   = CDC_NOTIFICATION_EPSIZE,
  						.Banks                  = 1,
  					},
57f5d0bc   dmohamed   FIX - Oublie de m...
78
79
  			},
  	};
41f6cbfd   dmohamed   first commit
80
  
1f4f9121   dmohamed   UPDATE:Inspiratio...
81
  
41f6cbfd   dmohamed   first commit
82
83
84
85
86
87
88
  /** Main program entry point. This routine contains the overall program flow, including initial
   *  setup of all components and the main program loop.
   */
  int main(void)
  {
  	SetupHardware();
  
1f4f9121   dmohamed   UPDATE:Inspiratio...
89
90
  	RingBuffer_InitBuffer(&USBtoUSART_Buffer, USBtoUSART_Buffer_Data, sizeof(USBtoUSART_Buffer_Data));
  	RingBuffer_InitBuffer(&USARTtoUSB_Buffer, USARTtoUSB_Buffer_Data, sizeof(USARTtoUSB_Buffer_Data));
41f6cbfd   dmohamed   first commit
91
92
  
  	LEDs_SetAllLEDs(LEDMASK_USB_NOTREADY);
1f4f9121   dmohamed   UPDATE:Inspiratio...
93
  	GlobalInterruptEnable();
41f6cbfd   dmohamed   first commit
94
95
96
  
  	for (;;)
  	{
1f4f9121   dmohamed   UPDATE:Inspiratio...
97
98
  		/* Only try to read in bytes from the CDC interface if the transmit buffer is not full */
  		if (!(RingBuffer_IsFull(&USBtoUSART_Buffer)))
57f5d0bc   dmohamed   FIX - Oublie de m...
99
  		{
1f4f9121   dmohamed   UPDATE:Inspiratio...
100
  			int16_t ReceivedByte = CDC_Device_ReceiveByte(&VirtualSerial_CDC_Interface);
57f5d0bc   dmohamed   FIX - Oublie de m...
101
  
1f4f9121   dmohamed   UPDATE:Inspiratio...
102
103
104
  			/* Store received byte into the USART transmit buffer */
  			if (!(ReceivedByte < 0))
  			  RingBuffer_Insert(&USBtoUSART_Buffer, ReceivedByte);
57f5d0bc   dmohamed   FIX - Oublie de m...
105
106
  		}
  
1f4f9121   dmohamed   UPDATE:Inspiratio...
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
  		uint16_t BufferCount = RingBuffer_GetCount(&USARTtoUSB_Buffer);
  		if (BufferCount)
  		{
  			Endpoint_SelectEndpoint(VirtualSerial_CDC_Interface.Config.DataINEndpoint.Address);
  
  			/* Check if a packet is already enqueued to the host - if so, we shouldn't try to send more data
  			 * until it completes as there is a chance nothing is listening and a lengthy timeout could occur */
  			if (Endpoint_IsINReady())
  			{
  				/* Never send more than one bank size less one byte to the host at a time, so that we don't block
  				 * while a Zero Length Packet (ZLP) to terminate the transfer is sent if the host isn't listening */
  				uint8_t BytesToSend = MIN(BufferCount, (CDC_TXRX_EPSIZE - 1));
  
  				/* Read bytes from the USART receive buffer into the USB IN endpoint */
  				while (BytesToSend--)
  				{
  					/* Try to send the next byte of data to the host, abort if there is an error without dequeuing */
  					if (CDC_Device_SendByte(&VirtualSerial_CDC_Interface,
  											RingBuffer_Peek(&USARTtoUSB_Buffer)) != ENDPOINT_READYWAIT_NoError)
  					{
  						break;
  					}
  
  					/* Dequeue the already sent byte from the buffer now we have confirmed that no transmission error occurred */
  					RingBuffer_Remove(&USARTtoUSB_Buffer);
  				}
  			}
  		}
  
  		/* Load the next byte from the USART transmit buffer into the USART if transmit buffer space is available */
  		if (Serial_IsSendReady() && !(RingBuffer_IsEmpty(&USBtoUSART_Buffer)))
  		  Serial_SendByte(RingBuffer_Remove(&USBtoUSART_Buffer));
41f6cbfd   dmohamed   first commit
139
140
141
142
143
144
145
146
147
  
  		CDC_Device_USBTask(&VirtualSerial_CDC_Interface);
  		USB_USBTask();
  	}
  }
  
  /** Configures the board hardware and chip peripherals for the demo's functionality. */
  void SetupHardware(void)
  {
1f4f9121   dmohamed   UPDATE:Inspiratio...
148
  #if (ARCH == ARCH_AVR8)
41f6cbfd   dmohamed   first commit
149
150
151
152
153
154
  	/* Disable watchdog if enabled by bootloader/fuses */
  	MCUSR &= ~(1 << WDRF);
  	wdt_disable();
  
  	/* Disable clock division */
  	clock_prescale_set(clock_div_1);
1f4f9121   dmohamed   UPDATE:Inspiratio...
155
  #endif
41f6cbfd   dmohamed   first commit
156
157
  
  	/* Hardware Initialization */
41f6cbfd   dmohamed   first commit
158
159
  	LEDs_Init();
  	USB_Init();
41f6cbfd   dmohamed   first commit
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
  }
  
  /** Event handler for the library USB Connection event. */
  void EVENT_USB_Device_Connect(void)
  {
  	LEDs_SetAllLEDs(LEDMASK_USB_ENUMERATING);
  }
  
  /** Event handler for the library USB Disconnection event. */
  void EVENT_USB_Device_Disconnect(void)
  {
  	LEDs_SetAllLEDs(LEDMASK_USB_NOTREADY);
  }
  
  /** Event handler for the library USB Configuration Changed event. */
  void EVENT_USB_Device_ConfigurationChanged(void)
  {
  	bool ConfigSuccess = true;
  
  	ConfigSuccess &= CDC_Device_ConfigureEndpoints(&VirtualSerial_CDC_Interface);
  
  	LEDs_SetAllLEDs(ConfigSuccess ? LEDMASK_USB_READY : LEDMASK_USB_ERROR);
  }
  
1f4f9121   dmohamed   UPDATE:Inspiratio...
184
185
  /** Event handler for the library USB Control Request reception event. */
  void EVENT_USB_Device_ControlRequest(void)
41f6cbfd   dmohamed   first commit
186
187
188
189
  {
  	CDC_Device_ProcessControlRequest(&VirtualSerial_CDC_Interface);
  }
  
57f5d0bc   dmohamed   FIX - Oublie de m...
190
191
192
193
194
195
196
  /** ISR to manage the reception of data from the serial port, placing received bytes into a circular buffer
   *  for later transmission to the host.
   */
  ISR(USART1_RX_vect, ISR_BLOCK)
  {
  	uint8_t ReceivedByte = UDR1;
  
1f4f9121   dmohamed   UPDATE:Inspiratio...
197
  	if ((USB_DeviceState == DEVICE_STATE_Configured) && !(RingBuffer_IsFull(&USARTtoUSB_Buffer)))
57f5d0bc   dmohamed   FIX - Oublie de m...
198
199
200
201
  	  RingBuffer_Insert(&USARTtoUSB_Buffer, ReceivedByte);
  }
  
  /** Event handler for the CDC Class driver Line Encoding Changed event.
41f6cbfd   dmohamed   first commit
202
203
204
   *
   *  \param[in] CDCInterfaceInfo  Pointer to the CDC class interface configuration structure being referenced
   */
57f5d0bc   dmohamed   FIX - Oublie de m...
205
  void EVENT_CDC_Device_LineEncodingChanged(USB_ClassInfo_CDC_Device_t* const CDCInterfaceInfo)
41f6cbfd   dmohamed   first commit
206
  {
57f5d0bc   dmohamed   FIX - Oublie de m...
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
  	uint8_t ConfigMask = 0;
  
  	switch (CDCInterfaceInfo->State.LineEncoding.ParityType)
  	{
  		case CDC_PARITY_Odd:
  			ConfigMask = ((1 << UPM11) | (1 << UPM10));
  			break;
  		case CDC_PARITY_Even:
  			ConfigMask = (1 << UPM11);
  			break;
  	}
  
  	if (CDCInterfaceInfo->State.LineEncoding.CharFormat == CDC_LINEENCODING_TwoStopBits)
  	  ConfigMask |= (1 << USBS1);
  
  	switch (CDCInterfaceInfo->State.LineEncoding.DataBits)
  	{
  		case 6:
  			ConfigMask |= (1 << UCSZ10);
  			break;
  		case 7:
  			ConfigMask |= (1 << UCSZ11);
  			break;
  		case 8:
  			ConfigMask |= ((1 << UCSZ11) | (1 << UCSZ10));
  			break;
  	}
  
1f4f9121   dmohamed   UPDATE:Inspiratio...
235
236
237
  	/* Keep the TX line held high (idle) while the USART is reconfigured */
  	PORTD |= (1 << 3);
  
57f5d0bc   dmohamed   FIX - Oublie de m...
238
239
240
241
242
243
244
245
246
247
248
249
  	/* Must turn off USART before reconfiguring it, otherwise incorrect operation may occur */
  	UCSR1B = 0;
  	UCSR1A = 0;
  	UCSR1C = 0;
  
  	/* Set the new baud rate before configuring the USART */
  	UBRR1  = SERIAL_2X_UBBRVAL(CDCInterfaceInfo->State.LineEncoding.BaudRateBPS);
  
  	/* Reconfigure the USART in double speed mode for a wider baud rate range at the expense of accuracy */
  	UCSR1C = ConfigMask;
  	UCSR1A = (1 << U2X1);
  	UCSR1B = ((1 << RXCIE1) | (1 << TXEN1) | (1 << RXEN1));
1f4f9121   dmohamed   UPDATE:Inspiratio...
250
251
252
  
  	/* Release the TX line after the USART has been reconfigured */
  	PORTD &= ~(1 << 3);
41f6cbfd   dmohamed   first commit
253
  }
1f4f9121   dmohamed   UPDATE:Inspiratio...