forked from wschutzer/xsvfduino
-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathports.cpp
More file actions
171 lines (151 loc) · 5.4 KB
/
ports.cpp
File metadata and controls
171 lines (151 loc) · 5.4 KB
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
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
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
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
/*******************************************************/
/* file: ports.c */
/* abstract: This file contains the routines to */
/* output values on the JTAG ports, to read */
/* the TDO bit, and to read a byte of data */
/* from the prom */
/* Revisions: */
/* 12/01/2008: Same code as before (original v5.01). */
/* Updated comments to clarify instructions.*/
/* Add print in setPort for xapp058_example.exe.*/
/*******************************************************/
#include "ports.h"
/*#include "prgispx.h"*/
// #include "stm32_gpio.h"
static uint32_t port_ = 0; /* Port vlaues */
/*BYTE *xsvf_data=0;*/
#if 1
inline void DLY()
{
asm(
"nop\n\t"
"nop\n\t"
"nop\n\t"
"nop\n\t"
"nop\n\t"
);
}
#else
#define DLY()
#endif
void initPorts()
{
pinMode(TCK, OUTPUT);
pinMode(TMS, OUTPUT);
pinMode(TDI, OUTPUT);
pinMode(TDO, INPUT_PULLUP);
}
/* setPort: Implement to set the named JTAG signal (p) to the new value (v).*/
/* if in debugging mode, then just set the variables */
void setPort(uint8_t p,uint8_t val)
{
/* Printing code for the xapp058_example.exe. You must set the specified
JTAG signal (p) to the new value (v). See the above, old Win95 code
as an implementation example. */
switch(p)
{
case TMS:
// g_iTMS = val;
if (val) port_ |= 1u << TMS_PIN; else port_ &= ~(1u << TMS_PIN);
break;
case TDI:
// g_iTDI = val;
if (val) port_ |= 1u << TDI_PIN; else port_ &= ~(1u << TDI_PIN);
break;
case TCK:
// g_iTCK = val;
if (val) port_ |= 1u << TCK_PIN; else port_ &= ~(1u << TCK_PIN);
#if 1
// uint32_t r = GPIOB->ODR;
// if (g_iTMS) r |= 1u << TMS_PIN; else r &= ~(1u << TMS_PIN);
// if (g_iTDI) r |= 1u << TDI_PIN; else r &= ~(1u << TDI_PIN);
// if (g_iTCK) r |= 1u << TCK_PIN; else r &= ~(1u << TCK_PIN);
// GPIOB->ODR = r;
//GPIOB->ODR = ( GPIOB->ODR & ~( (1u << TMS_PIN) | (1u << TDI_PIN) | (1u << TCK_PIN) ) ) | port_;
GPIOB->regs->ODR = ( GPIOB->regs->ODR & ~( (1u << TMS_PIN) | (1u << TDI_PIN) | (1u << TCK_PIN) ) ) | port_;
#else
digitalWrite( TMS, g_iTMS );
digitalWrite( TDI, g_iTDI );
digitalWrite( TCK, g_iTCK );
#endif
DLY();
break;
}
}
/* toggle tck LH. No need to modify this code. It is output via setPort. */
void pulseClock()
{
setPort(TCK,0); /* set the TCK port to low */
setPort(TCK,1); /* set the TCK port to high */
}
/* readByte: Implement to source the next byte from your XSVF file location */
/* read in a byte of data from the prom */
void readByte(uint8_t *data)
{
*data = read_data();
/**data=*xsvf_data++;*/
}
/* waitTime: Implement as follows: */
/* REQUIRED: This function must consume/wait at least the specified number */
/* of microsec, interpreting microsec as a number of microseconds.*/
/* REQUIRED FOR SPARTAN/VIRTEX FPGAs and indirect flash programming: */
/* This function must pulse TCK for at least microsec times, */
/* interpreting microsec as an integer value. */
/* RECOMMENDED IMPLEMENTATION: Pulse TCK at least microsec times AND */
/* continue pulsing TCK until the microsec wait */
/* requirement is also satisfied. */
void waitTime(uint32_t microsec)
{
#if 1
static long tckCyclesPerMicrosec = 1; /* must be at least 1 */
long tckCycles = microsec * tckCyclesPerMicrosec;
long i;
/* This implementation is highly recommended!!! */
/* This implementation requires you to tune the tckCyclesPerMicrosec
variable (above) to match the performance of your embedded system
in order to satisfy the microsec wait time requirement. */
for ( i = 0; i < tckCycles; ++i )
{
pulseClock();
}
#endif
#if 0
uint32_t t0 = micros();
while( micros()-t0 < microsec )
{
pulseClock();
}
#endif
#if 0
/* Alternate implementation */
/* For systems with TCK rates << 1 MHz; Consider this implementation. */
/* This implementation does not work with Spartan-3AN or indirect flash
programming. */
if ( microsec >= 50L )
{
/* Make sure TCK is low during wait for XC18V00/XCFxxS */
/* Or, a running TCK implementation as shown above is an OK alternate */
setPort( TCK, 0 );
/* Use Windows Sleep(). Round up to the nearest millisec */
_sleep( ( microsec + 999L ) / 1000L );
}
else /* Satisfy FPGA JTAG configuration, startup TCK cycles */
{
for ( i = 0; i < microsec; ++i )
{
pulseClock();
}
}
#endif
#if 0
/* Alternate implementation */
/* This implementation is valid for only XC9500/XL/XV, CoolRunner/II CPLDs,
XC18V00 PROMs, or Platform Flash XCFxxS/XCFxxP PROMs.
This implementation does not work with FPGAs JTAG configuration. */
/* Make sure TCK is low during wait for XC18V00/XCFxxS PROMs */
/* Or, a running TCK implementation as shown above is an OK alternate */
setPort( TCK, 0 );
/* Use Windows Sleep(). Round up to the nearest millisec */
u_sleep( ( microsec + 999L ) / 1000L );
#endif
}