Magstripe.c 7.49 KB
Newer Older
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
/*
             LUFA Library
     Copyright (C) Dean Camera, 2009.
              
  dean [at] fourwalledcubicle [dot] com
      www.fourwalledcubicle.com
*/

/*
  Copyright 2009  Denver Gingerich (denver [at] ossguy [dot] com)
  Copyright 2009  Dean Camera (dean [at] fourwalledcubicle [dot] com)

  Permission to use, copy, modify, and distribute this software
  and its documentation for any purpose and without fee is hereby
  granted, 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.

  The author disclaim all warranties with regard to this
  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.
*/
 
32
33
34
35
36
37
/** \file
 *
 *  Main source file for the MagStripe reader program. This file contains the main tasks of
 *  the project and is responsible for the initial application hardware configuration.
 */
 
38
39
#include "Magstripe.h"

40
41
42
/** Bit buffers to hold the read bits for each of the three magnetic card tracks before they are transmitted
 *  to the host as keyboard presses.
 */
43
44
45
46
BitBuffer_t TrackDataBuffers[TOTAL_TRACKS];

/** Pointer to the current track buffer being sent to the host. */
BitBuffer_t* CurrentTrackBuffer = &TrackDataBuffers[TOTAL_TRACKS];
47

48
49
50
51
/** LUFA HID Class driver interface configuration and state information. This structure is
 *  passed to all HID Class driver functions, so that multiple instances of the same class
 *  within a device can be differentiated from one another.
 */
52
USB_ClassInfo_HID_Device_t Keyboard_HID_Interface =
53
	{
Dean Camera's avatar
Dean Camera committed
54
55
56
57
58
59
60
		.Config =
			{
				.InterfaceNumber         = 0,

				.ReportINEndpointNumber  = KEYBOARD_EPNUM,
				.ReportINEndpointSize    = KEYBOARD_EPSIZE,
			},
61
	};
62

63
64
65
/** Main program entry point. This routine contains the overall program flow, including initial
 *  setup of all components and the main program loop.
 */
66
67
68
69
int main(void)
{
	SetupHardware();
	
70
	for (uint8_t Buffer = 0; Buffer < TOTAL_TRACKS; Buffer++)
71
	  BitBuffer_Init(&TrackDataBuffers[Buffer]);
72

73
74
75
76
	for (;;)
	{
		if (Magstripe_GetStatus() & MAG_CARDPRESENT)
		  ReadMagstripeData();
77

Dean Camera's avatar
Dean Camera committed
78
		HID_Device_USBTask(&Keyboard_HID_Interface);
79
80
81
		USB_USBTask();
	}
}
82

83
/** Configures the board hardware and chip peripherals for the demo's functionality. */
84
void SetupHardware(void)
85
86
87
88
89
{
	/* Disable watchdog if enabled by bootloader/fuses */
	MCUSR &= ~(1 << WDRF);
	wdt_disable();

90
91
	/* Disable clock division */
	clock_prescale_set(clock_div_1);
92
93
94
95

	/* Hardware Initialization */
	Magstripe_Init();
	USB_Init();
96
97
98
99
100
101

	/* Millisecond timer initialization, with output compare interrupt enabled for the idle timing */
	OCR0A  = ((F_CPU / 64) / 1000);
	TCCR0A = (1 << WGM01);
	TCCR0B = ((1 << CS01) | (1 << CS00));
	TIMSK0 = (1 << OCIE0A);
102
103
}

104
105
106
/** Determines if a card has been inserted, and if so reads in each track's contents into the bit buffers
 *  until they are read out to the host as a series of keyboard presses.
 */
107
void ReadMagstripeData(void)
108
{
109
110
111
112
113
114
115
116
	/* Arrays to hold the buffer pointers, clock and data bit masks for the separate card tracks */
	const struct
	{
		uint8_t ClockMask;
		uint8_t DataMask;	
	} TrackInfo[] = {{MAG_T1_CLOCK, MAG_T1_DATA},
	                 {MAG_T2_CLOCK, MAG_T2_DATA},
	                 {MAG_T3_CLOCK, MAG_T3_DATA}};
117

118
119
120
121
122
	uint8_t Magstripe_Prev = 0;
	uint8_t Magstripe_LCL  = Magstripe_GetStatus();

	while (Magstripe_LCL & MAG_CARDPRESENT)
	{
123
		for (uint8_t Track = 0; Track < TOTAL_TRACKS; Track++)
124
125
126
127
128
		{
			bool DataPinLevel      = ((Magstripe_LCL & TrackInfo[Track].DataMask) != 0);
			bool ClockPinLevel     = ((Magstripe_LCL & TrackInfo[Track].ClockMask) != 0);
			bool ClockLevelChanged = (((Magstripe_LCL ^ Magstripe_Prev) & TrackInfo[Track].ClockMask) != 0);
		
129
			/* Sample data on rising clock edges from the card reader */
130
131
132
133
134
135
136
			if (ClockPinLevel && ClockLevelChanged)
			  BitBuffer_StoreNextBit(&TrackDataBuffers[Track], DataPinLevel);
		}

		Magstripe_Prev = Magstripe_LCL;
		Magstripe_LCL  = Magstripe_GetStatus();
	}
137
138
	
	CurrentTrackBuffer = &TrackDataBuffers[0];
139
140
}

141
/** Event handler for the library USB Configuration Changed event. */
142
void EVENT_USB_ConfigurationChanged(void)
143
{
Dean Camera's avatar
Dean Camera committed
144
	HID_Device_ConfigureEndpoints(&Keyboard_HID_Interface);
145
146
}

147
/** Event handler for the library USB Unhandled Control Packet event. */
148
void EVENT_USB_UnhandledControlPacket(void)
149
{
Dean Camera's avatar
Dean Camera committed
150
	HID_Device_ProcessControlPacket(&Keyboard_HID_Interface);
151
152
}

153
/** Timer 0 CTC ISR, firing once each millisecond to keep track of elapsed idle time in the HID interface. */
154
ISR(TIMER0_COMPA_vect, ISR_BLOCK)
155
{
156
	HID_Device_MillisecondElapsed(&Keyboard_HID_Interface);
157
158
}

159
160
/** HID Class driver callback function for the creation of a HID report for the host.
 *
161
162
163
 *  \param[in] HIDInterfaceInfo  Pointer to the HID interface structure for the HID interface being referenced
 *  \param[in,out] ReportID      Report ID requested by the host if non-zero, otherwise callback should set to the generated report ID
 *  \param[out] ReportData       Pointer to the preallocated report buffer where the created report should be stored
164
165
166
 *
 *  \return Number of bytes in the created report
 */
167
uint16_t CALLBACK_HID_Device_CreateHIDReport(USB_ClassInfo_HID_Device_t* const HIDInterfaceInfo, uint8_t* const ReportID, void* ReportData)
168
{
169
170
	USB_KeyboardReport_Data_t* KeyboardReport = (USB_KeyboardReport_Data_t*)ReportData;

171
	static bool IsKeyReleaseReport;
172

173
	/* Key reports must be interleaved with key release reports, or repeated keys will be ignored */
174
	IsKeyReleaseReport = !IsKeyReleaseReport;
175

176
	if ((IsKeyReleaseReport) || (CurrentTrackBuffer == &TrackDataBuffers[TOTAL_TRACKS]))
177
	{
178
		/* No more data to send, or key release report between key presses */
179
		KeyboardReport->KeyCode = KEY_NONE;
180
	}
181
	else if (!(CurrentTrackBuffer->Elements))
182
	{
183
		/* End of current track, send an enter press and change to the next track's buffer */
184
		KeyboardReport->KeyCode = KEY_ENTER;
185
		CurrentTrackBuffer++;
186
187
	}
	else
188
	{
189
190
		/* Still data in the current track; convert next bit to a 1 or 0 keypress */
		KeyboardReport->KeyCode = BitBuffer_GetNextBit(CurrentTrackBuffer) ? KEY_1 : KEY_0;
191
192
	}
	
193
	return sizeof(USB_KeyboardReport_Data_t);
194
195
}

196
197
/** HID Class driver callback function for the processing of a received HID report from the host.
 *
198
199
200
201
 *  \param[in] HIDInterfaceInfo  Pointer to the HID interface structure for the HID interface being referenced
 *  \param[in] ReportID          Report ID of the received report from the host
 *  \param[in] ReportData        Pointer to the report buffer where the received report is stored
 *  \param[in] ReportSize        Size in bytes of the report received from the host
202
 */
203
204
void CALLBACK_HID_Device_ProcessHIDReport(USB_ClassInfo_HID_Device_t* const HIDInterfaceInfo, const uint8_t ReportID,
                                          const void* ReportData, const uint16_t ReportSize)
205
{
206
	// Unused (but mandatory for the HID class driver) in this demo, since there are no Host->Device reports
207
}