summaryrefslogtreecommitdiff
path: root/src/app/transactiontest/util.S
blob: c0b1ebcbe459538539cbabf81b110566754b2856 (plain)
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
.global asm_save_all
.global asm_load_all
.global asm_load_mem

stack_backup:
    .space 2048

sp_backup:
    .space 2

asm_save_all:
    dint
    .irp reg,4,5,6,7,8,9,10,11
    push r\reg
    .endr

    mov r1, &sp_backup

    mov #1c00h, r10
    mov #stack_backup, r11

save_sram_word:
    mov @r10+, 0(r11)
    add #2, r11
    cmp #1c00h+2048, r10
    jlo save_sram_word

    pop r11
    pop r10
    add #12, r1
    eint
    ret

asm_load_all:
    dint

    ; restore SRAM from backup
    mov #stack_backup, r10
    mov #1c00h, r11
load_sram_word:
    mov @r10+, 0(r11)
    add #2, r11
    cmp #1c00h+2048, r11
    jlo load_sram_word

    ; restore stack pointer
    mov &sp_backup, r1

    ; restore registers
    .irp reg,11,10,9,8,7,6,5,4
    pop r\reg
    .endr

    eint

    ; The return address on the stack has been restored from FRAM backup
    ; -> execution will continue where asm_save_all was called
    ret

asm_load_mem:
    dint
    ; save return address in r4
    mov @r1, r4

    ; restore SRAM from backup
    mov #stack_backup, r10
    mov #1c00h, r11
load_sram_word2:
    mov @r10+, 0(r11)
    add #2, r11
    cmp #1c00h+2048, r11
    jlo load_sram_word2

    ; restore stack pointer
    mov &sp_backup, r1

    ; restore old register contents
    .irp reg,11,10,9,8,7,6,5
    pop r\reg
    .endr

    ; load return address, that is, the address this function was called from.
    ; This is not the address which called asm_save_all -- we only want to restore memory contents, not re-execute everything
    mov r4, 2(r1)

    ; load remaining register content
    pop r4

    eint
    ret