-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathuart-spm.asm
384 lines (288 loc) · 9.84 KB
/
uart-spm.asm
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
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
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
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
.include "tn13adef.inc"
.dseg
.org SRAM_START
msg: .byte 34 ; allocate RAM space for UART data
;***** Pin definitions
.equ RxD = 3 ;Receive pin is PB3
.equ TxD = 4 ;Transmit pin is PB4
;***** UART Rx/Tx delays
.equ tx_delay = 163
;Number of stop bits (1, 2, ...)
.equ sb = 1
;***** Global register variables
.def bitcnt = R16 ;bit counter
.def temp = R17 ;temporary storage register
.def Txbyte = R18 ;Data to be transmitted
.def RXbyte = R19 ;Received data
;r20 flag - bit 0: rx flag, bit 1: mode flag
;r21 - used by interrupt timer because attiny13a prescaler is not enough
.cseg
.org 0
rjmp init ;powerup / reset routine
nop ; IRQ0
rjmp PCINT ; PCINT0 (pin INT0 to INT5) PCMSK, MCUCR
rjmp TOV ; Timer0 Overflow
nop ; EEPROM Ready
nop ; Analog Comparator
nop ; Timer0 CompareA
nop ; Timer 0 compare B
nop ; watchdog
nop ; ADC conversion
;***** Program Execution Starts Here
init:
; flash end: 512 (dec) 0x200 (hex)
;note: flash is word addressed, so 1KB flash , end address is 512 (x16 bits)
;important: flash data to use with SPM MUST be at start of a page, so position MUST be a multiple of the pagesize (16 for attiny13a)
; e.g. (dec) 128, 256 etc
.org 0x130
;304 (dec) in words, so 608(dec) in bytes, so 608 bytes max allowed for code
;392 bytes / 196 words reserved for our flash_data
;max 12 pages
flash_data: ; reserve FLASH memory for data storage
.dw 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0
.dw 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0
.dw 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0
.dw 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0
.dw 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0
.dw 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0
.dw 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0
.dw 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0
.dw 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0
;.dw 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0
.org 0x000A ; First address after interrupt vector table, see datasheet 9.1
sbi PORTB,TxD ;Init port pins
sbi DDRB,TxD
;init for receiving
ldi r16,0b00100000 ;GIMSK = (1<<PCIE); Enable External Interrupts
out GIMSK,r16
sbi PCMSK,RxD ;PCMSK = (1<<RxD); Enable accorded Interrupt (PCINT3)
;TIMER
ldi r16,0b00000000
out TCCR0A,r16 ; normal timer 0-255
IN R16, TCCR0B ;clk I/O /1024 (From prescaler)
ORI R16, (1<<CS02) | (0<<CS01) | (1<<CS00)
OUT TCCR0B, R16
;clr r16 - shouldnt be needed as bitcnt is loaded before use in routines
clr r21 - not needed as 0 on powerup
;reset flash pointer if not already done
ldi ZL,LOW(2*flash_data)
ldi ZH,HIGH(2*flash_data)
sei; Allow Interrupts
;clr r20 ; rx flag - not needed as 0 on powerup
;############################################
;;########### MAIN LOOP #####################
;############################################
main_loop:
receiving_mode:
sbrs r20,1 ;check mode flag
rjmp not_receiving
sbrs r20,0;check rx flag
rjmp receiving_mode
;echo what was sent
;rcall UART_Send
;write the 32 bytes received to flash -if less recd it will run over trash in RAM til 32 bytes are reached
rcall write_page
;send flash contents to verify
rcall UART_Send_Flash
cbr r20,1 ; clear rx flag
wait_for_idle:
sbis PINB,RxD ;Wait for idle bit
rjmp wait_for_idle
sbi PCMSK,RxD ;re-enable PCINT once idle bit detected
rjmp receiving_mode
not_receiving:
;stuff we do outside the UART receiving mode:
rjmp main_loop
;############################################
;;########### MAIN LOOP #####################
;############################################
;***************************************************************************
;* Timer interrupt routine
;***************************************************************************
TOV:
inc r21 ;increase r21 - used in addition to prescaler of 1024
cpi r21,100 ; if r21 is at ?, then run the code below
brne end_t0 ; if not, exit interrupt
clr r20 ;clear both rx and mode flags (rx should already be clear)
out TIMSK0, r20 ;disable timer interrupt (use fact r20 is already cleared)
end_t0:
reti
;***************************************************************************
;* "getchar" - UART Interrupt routine
;***************************************************************************
PCINT:
in r0, 0x3F ;Store SREG
push r0
ldi YL,LOW(msg) ; initialize pointer
ldi YH,HIGH(msg) ; to tx msg address
;if we just went into SPM mode
cpi r20,0 ;all flags should still be zero
brne skip_spm_init ;if we're already in SPM mode, skip this
;initialize flash pointer
ldi ZL,LOW(2*flash_data)
ldi ZH,HIGH(2*flash_data)
skip_spm_init:
cbi PCMSK,RxD ; turn off PCINT
getchar:
ldi bitcnt,9 ;8 data bit + 1 stop bit
getchar1:
rcall UART_delay ;0.5 bit delay
getchar2:
rcall UART_delay ;1 bit delay
rcall UART_delay
clc ;clear carry
sbic PINB,RxD ;if RX pin high
sec ;
dec bitcnt ;If bit is stop bit
breq getchar3 ; return
;else
ror Rxbyte ; shift bit into Rxbyte
rjmp getchar2 ; go get next
getchar3:
st Y+,Rxbyte
cpi Rxbyte,0x0D ; check for carriage return byte
breq exitrx ; if yes then end receiving
;if not, do below:
ldi bitcnt,9 ;8 data bit + 1 stop bit
sb_wait:
sbic PINB,RxD ;Wait for start bit
rjmp sb_wait
rjmp getchar1
exitrx:
ldi Rxbyte,0x0A ;add linebreak
st Y+,Rxbyte
clr Rxbyte ; null terminated string
st Y,Rxbyte
sbr r20, 3 ;set both rx and mode flags - bits 0 (rx) and 1 (mode) in r20 , binary/dec 11/3
;enable timer0 overflow interrupt
ldi r16, 0b00000010
OUT TIMSK0, R16
pop r0 ; Restore SREG
out 0x3F, r0
reti
;***************************************************************************
;* "putchar"
;***************************************************************************
putchar:
ldi bitcnt,9+sb ;1+8+sb (sb is # of stop bits)
com Txbyte ;Inverte everything
sec ;Start bit
putchar0:
brcc putchar1 ;If carry set
cbi PORTB,TxD ; send a '0'
rjmp putchar2 ;else
putchar1:
sbi PORTB,TxD ; send a '1'
nop
putchar2:
rcall UART_delay ;One bit delay
rcall UART_delay
lsr Txbyte ;Get next bit
dec bitcnt ;If not all bit sent
brne putchar0 ; send next
;else
ret ; return
;***************************************************************************
;* Send series of bytes from RAM
;***************************************************************************
UART_Send:
ldi YL,LOW(msg) ; initialize pointer
ldi YH,HIGH(msg) ; to tx msg address
cli ;Prevent Global Interrupts
TX_Byte:
ld r18, Y+ ;Load data byte for sending
cpi r18, 0 ;Compare with ZERO
breq Exit_Transmit ; Exit if equal, EXIT transmitting
rcall putchar
rjmp TX_Byte ;jump to next sending byte
Exit_Transmit:
sei ;Allow Global Interrupts
ret
;***************************************************************************
;* send first R22 bytes of flash_data
;***************************************************************************
UART_Send_Flash:
push ZL
push ZH
ldi ZL,LOW(2*flash_data) ;load the starting page address into Z
ldi ZH,HIGH(2*flash_data)
ldi r22, 255 ; first 255 bytes
cli ;Prevent Global Interrupts
TX_B:
dec r22
breq Exit_Tx ; Exit if equal, EXIT transmitting
lpm r18,Z+ ;Load data byte for sending
rcall putchar
rjmp TX_B ;jump to next sending byte
Exit_Tx:
sei ;Allow Global Interrupts
pop ZH
pop ZL
ret
;***************************************************************************
;*
;* "UART_delay"
;*
;* This delay subroutine generates the required delay between the bits when
;* transmitting and receiving bytes. The total execution time is set by the
;* constant "b":
;*
;* 3·b + 7 cycles (including rcall and ret)
;***************************************************************************
.equ b = 163 ;9600 bps @ 9.6 MHz crystal
UART_delay:
ldi temp,b ;1c
UART_delay1:
dec temp ;1c
brne UART_delay1 ;1c
ret
;***************************************************************************
;*
;* SPM - Store to Program Memory (flash) from RAM (first 32 bytes)
;*
;***************************************************************************
;flash is arranged as 512x16 bits
;for attiny13A PAGESIZE = 16 (words) so PAGESIZEB = 32
;PAGESIZEB = PAGESIZE*2 ;PAGESIZEB is page size in BYTES, not words // 1 word = 2 bytes
write_page:
in r19, SREG ;store and then restore SREG - note: r19 is also used as RxByte by uart routine
cli
ldi YL,LOW(msg) ;initialize Y pointer
ldi YH,HIGH(msg) ;to ram
ldi r16,0b00000011 ;this is the page erase command
rcall do_spm
;transfer data from RAM to Flash page buffer
ldi r17, 16 ;init loop variable - temp r17 also used by uart delay routine
ldi r16,0b00000001 ;load the page buffer with r0:r1
wrloop:
ld r0,Y+
ld r1,Y+
rcall do_spm
adiw ZH:ZL, 2 ;since one word is being programmed, increment twice
dec r17 ; same as subi r17,2 when r17 is 32
brne wrloop
;execute page write
sbiw ZH:ZL, 32 ; restore pointer back to initial pos for page write
ldi r16, 0b00000101 ;page write command
rcall do_spm
adiw ZH:ZL, 32 ; move pointer back to last pos for the next uart spm to continue from there
out SREG, r19
sei
ret
do_spm:
;input: spmcrval determines SPM action
;disable interrupts if enabled, store status
;in r19, SREG
;cli
;check for previous SPM complete
;wait:
; in r18, SPMCSR
; sbrc r18, SPMEN
; rjmp wait
;SPM timed sequence
out SPMCSR, r16
spm
;restore SREG (to enable interrupts if originally enabled)
;out SREG, r19
;sei
ret