summaryrefslogtreecommitdiffstats
path: root/source/opendingux/port-asm.S
blob: 127b40a680e02a7d0ce1cd59eb7ac0061cbdbf0c (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
91
92
93
94
95
96
97
98
99
.global ReGBA_MakeCodeVisible

.set noat
.set noreorder

.ent ReGBA_MakeCodeVisible

# Register assignment:
# $4 = parameter #1: void* Code
# $5 = parameter #2: unsigned int CodeLength
# Clobbered: $1, $2, $4, $5
ReGBA_MakeCodeVisible:
#ifdef MIPS_32R2
  rdhwr $1, $1            # read SYNCI_Step into $1
  beq   $1, $0, 2f        # no need to use SYNCI? return
  addiu $2, $1, -1

  beq   $5, $0, 2f
  add   $5, $4, $5        # $5 is now End, derived from CodeLength (delay)
  nor   $2, $2, $0        # $2 is now the mask to use to round $4 down
  and   $4, $4, $2        # Round $4 down to the cache line containing it

  # The rounding down was necessary above in order to allow a partial
  # cache line near the end of the code to be flushed. Otherwise, it may
  # have been ignored after flushing one cache line's worth of bytes.
  # Consider:
  # Cache line 1 |                [flushflushflushf]
  # Cache line 2 | lushflushflu
  # Cache line 2 does not start a line to be flushed, so it isn't flushed.
  # The rounding down is required to make it this instead:
  # Cache line 1 |[padpadpadpadpad flushglushflushf]
  # Cache line 2 |[lushflushflu                    ]
  
  # Now git 'er done.
1:
.set mips32r2
  synci ($4)              # Combined Data Writeback-Instruction Invalidate (R2)
.set mips0
  addu  $4, $4, $1        # go to the next cache line
  sltu  $2, $4, $5        # if Code < End
  bne   $2, $0, 1b        #   goto 1
  nop                     # cannot delay usefully here

  sync                    # guard against memory hazards

2:
.set mips32r2
  jr.hb $ra               # return to caller, while guarding against
                          #   instruction hazards
.set mips0
  nop                     # cannot delay usefully here
#else
  # Issue the cacheflush kernel call. Preserve registers first.
  addiu $sp, $sp, -68
  sw    $ra, 0($sp)
  sw    $3, 4($sp)
  sw    $6, 8($sp)                # preserve the GBA PC register
  sw    $7, 12($sp)
  sw    $8, 16($sp)
  sw    $9, 20($sp)
  sw    $10, 24($sp)
  sw    $11, 28($sp)
  sw    $12, 32($sp)
  sw    $13, 36($sp)
  sw    $14, 40($sp)
  sw    $15, 44($sp)
  sw    $24, 48($sp)
  sw    $25, 52($sp)
  sw    $18, 56($sp)
  sw    $28, 60($sp)
  sw    $30, 64($sp)

  # Pass $4 and $5 through to the call.
  jal   cacheflush
  ori   $6, $0, 3                 # parameter #3: BCACHE (1|2) (branch delay)

  lw    $ra, 0($sp)               # restore the return address first, because
  lw    $3, 4($sp)                # restoring it last would cause an interlock
  lw    $6, 8($sp)                # $6 = GBA PC, also restore this early
  lw    $7, 12($sp)
  lw    $8, 16($sp)
  lw    $9, 20($sp)
  lw    $10, 24($sp)
  lw    $11, 28($sp)
  lw    $12, 32($sp)
  lw    $13, 36($sp)
  lw    $14, 40($sp)
  lw    $15, 44($sp)
  lw    $24, 48($sp)
  lw    $25, 52($sp)
  lw    $18, 56($sp)
  lw    $28, 60($sp)
  lw    $30, 64($sp)

  jr    $ra                       # return to the caller
  addiu $sp, $sp, 68              # adjust the stack back (branch delay)
#endif

.end ReGBA_MakeCodeVisible