forked from melonDS-emu/melonDS
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathbios_common.S
executable file
·1206 lines (961 loc) · 35.1 KB
/
bios_common.S
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
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
# Custom NDS ARM7/ARM9 BIOS replacement
# Copyright (c) 2013, Gilead Kutnick
# All rights reserved.
#
# Redistribution and use in source and binary forms, with or without
# modification, are permitted provided that the following conditions are met:
#
# 1) Redistributions of source code must retain the above copyright notice,
# this list of conditions and the following disclaimer.
# 2) Redistributions in binary form must reproduce the above copyright notice,
# this list of conditions and the following disclaimer in the documentation
# and/or other materials provided with the distribution.
#
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
# AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
# ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
# LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
# CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
# SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
# INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
# CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
# ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
# POSSIBILITY OF SUCH DAMAGE.
.section .text
.org 0x00000000
// Vector table
b boot_handler // 0x00 Reset
b unhandled_exception // 0x04 Undefined
b swi_handler // 0x08 SWI
b unhandled_exception // 0x0C Abort Prefetch
b unhandled_exception // 0x10 Abort Data
b unhandled_exception // 0x14 Reserved
b interrupt_handler // 0x18 IRQ
b unhandled_exception // 0x1C FIQ
#ifdef BIOS_ARM9
// ARM9 BIOS has a logo here (0x9C bytes), we don't want to include the logo
// but we'll leave a space for it.
.fill 0xe0, 0x1, 0x0
#endif
#ifdef BIOS_ARM7
// ARM7 BIOS has encryption stuff at 0x30, we don't want to include it but we'll
// leave a space for it.
.fill 0x1058, 0x1, 0x0
#endif
// TODO: This needs to be implemented. Not needed for emulators that HLE bootup.
boot_handler:
0:
b 0b
// Not sure what this should do, it's probably best to be pretty visible though.
unhandled_exception:
0:
b 0b
#ifdef BIOS_ARM7
#define swi_label(function) .word function
#define swi_label_arm7_only(function) swi_label(function)
#else
#define swi_label_arm7_only(function) swi_label(swi_invalid)
#endif
#ifdef BIOS_ARM9
#define swi_label(function) .word (function + 0xFFFF0000)
#define swi_label_arm9_only(function) swi_label(function)
#else
#define swi_label_arm9_only(function) swi_label(swi_invalid)
#endif
// SWI calling convention:
// Parameters are passed in via r0 - r3
// Called SWI can modify r0 - r3 (and return things here) and r4, r12, and r14.
// They can't modify anything else.
#define swi_comment r12
#define saved_spsr r4
#define modified_spsr r4
swi_handler:
// Save these as temporaries
stmdb sp!, { r4, r12, lr }
// get SPSR and enter system mode, interrupts on
mrs saved_spsr, SPSR
// This must be stacked and not just saved, because otherwise SWI won't
// be reentrant, which can happen if you're waiting for interrupts and the
// interrupt handler triggers the SWI.
stmdb sp!, { saved_spsr }
and modified_spsr, saved_spsr, #0x80
orr modified_spsr, modified_spsr, #0x1F
// Load comment from SWI instruction, which indicates which SWI
// to use.
ldrb swi_comment, [ lr, #-2 ]
msr CPSR_fc, modified_spsr
// We have to now save system-mode lr register as well
stmdb sp!, { lr }
// Not sure if this should be here or not, but it probably doesn't
// hurt, and is better than flooding the table with 256 entries..
// This will move in the known entry of an invalid SWI.
cmp swi_comment, #0x20
movge swi_comment, #0x01
// Branch to SWI handler
ldr pc, [ pc, swi_comment, lsl #2 ]
nop
// SWI table begins here
// If there's no entry just go straight to swi_complete
swi_label(swi_soft_reset) // 00
swi_label(swi_invalid) // 01
swi_label(swi_invalid) // 02
swi_label(swi_wait_by_loop) // 03
swi_label(swi_interrupt_wait) // 04
swi_label(swi_vblank_interrupt_wait) // 05
swi_label(swi_halt) // 06
swi_label_arm7_only(swi_stop) // 07
swi_label_arm7_only(swi_sound_bias) // 08
swi_label(swi_divide) // 09
swi_label(swi_invalid) // 0A
swi_label(swi_cpu_set) // 0B
swi_label(swi_cpu_fast_set) // 0C
swi_label(swi_sqrt) // 0D
swi_label(swi_get_crc16) // 0E
swi_label(swi_is_debugger) // 0F
swi_label(swi_bit_unpack) // 10
swi_label(swi_lz77_decompress_wram) // 11
swi_label(swi_lz77_decompress_vram) // 12
swi_label(swi_huffman_decompress) // 13
swi_label(swi_runlength_decompress_wram) // 14
swi_label(swi_runlength_decompress_vram) // 15
swi_label_arm9_only(swi_diff_8bit_unfilter_wram) // 16
swi_label(swi_invalid) // 17
swi_label_arm9_only(swi_diff_16bit_unfilter) // 18
swi_label(swi_invalid) // 19
swi_label_arm7_only(swi_get_sine_table) // 1A
swi_label_arm7_only(swi_get_pitch_table) // 1B
swi_label_arm7_only(swi_get_volume_table) // 1C
swi_label_arm7_only(swi_get_boot_procs) // 1D
swi_label(swi_custom_halt_post) // 1F
swi_invalid:
// This just passes through to completion.
// SWI returns here
swi_complete:
// Restore system mode lr
ldmia sp!, { lr }
// Go back to supervisor mode to get back to that stack
mov modified_spsr, #0xD3
msr CPSR_fc, modified_spsr
// SPSR has to be restored because the transition to system mode broke it
ldmia sp!, { saved_spsr }
msr SPSR, saved_spsr
// Restore stuff we saved
ldmia sp!, { r4, r12, lr }
// Return from exception handler
movs pc, lr
padding_a:
.word 0x0
swi_halt:
#ifdef BIOS_ARM7
mov r0, #0x04000000
mov r2, #0x80
strb r2, [ r0, #0x301 ]
#else
mov r0, #0x0
mcr p15, 0, r0, cr7, cr0, 4
#endif
b swi_complete
swi_wait_by_loop:
0:
subs r0, r0, #1
bgt 0b
b swi_complete
#define check_immediately r0
#define irq_wait_mask r1
#ifdef BIOS_ARM7
#define irq_flag_base r3
#else
#define irq_flag_base r2
#endif
#define io_base r3
#define const_0x1 r12
#define irq_flags r0
interrupt_check:
#ifdef BIOS_ARM9
// Get DTCM base
mrc p15, 0, irq_flag_base, cr9, cr1, 0
bic irq_flag_base, irq_flag_base, #0xFF
// Software IRQ flag is at DTCM[0x3FF8]
add irq_flag_base, irq_flag_base, #0x4000
#endif
// Load software IRQ flag
ldr irq_flags, [ irq_flag_base, #-8 ]
// Set IME (0x04000208) to 0
str io_base, [ io_base, #0x208 ]
// Check if IRQs were risen, to see if the loop can exit
tst irq_wait_mask, irq_flags
// Clear IRQs that were risen and write back
bic irq_flags, irq_flags, irq_wait_mask
str irq_flags, [ irq_flag_base, #-8 ]
mov const_0x1, #0x1
// Set IME (0x04000208) to 1 and return
str const_0x1, [ io_base, #0x208 ]
bx lr
#define halt_value r0
swi_vblank_interrupt_wait:
// Check immediately for VBLANK interrupt
mov check_immediately, #1
mov irq_wait_mask, #1
// Fall through
swi_interrupt_wait:
mov io_base, #0x4000000
// See if we should return immediately or halt
cmp check_immediately, #0
blne interrupt_check
// Perform this loop until the interrupt is risen
0:
#ifdef BIOS_ARM9
// Halt ARM9 via coprocessor instruction
mov halt_value, #0
mcr p15, 0, halt_value, cr7, cr0, 4
#else
mov halt_value, #0x80
// Set HALTCNT to 0x80
strb halt_value, [ io_base, #0x301 ]
#endif
bl interrupt_check
beq 0b
b swi_complete
swi_interrupt_check_first:
// Check for IRQ
bl interrupt_check
// If set exit.
bne swi_complete
// If not wait for interrupt.
b 0b
#ifdef BIOS_ARM7
swi_stop:
mov r0, #0x04000000
mov r1, #0xC0
strb r1, [ r0, #0x301 ]
b swi_complete
#define bias_ptr r0
#define bias_value r1
swi_sound_bias:
mov bias_ptr, #0x4000000
add bias_ptr, bias_ptr, #0x500
ldr bias_value, [ bias_ptr, #0x4 ]
cmp bias_value, #0
movne bias_value, #0x200
str bias_value, [ bias_ptr, #0x4 ]
// TODO: Needs to add delay
b swi_complete
#endif
#define numerator r0
#define denominator r1
#define accumulator r2
#define current_bit r3
#define numerator_signed r12
#define denominator_signed r3
#define sign_flip r12
#define result r0
#define remainder r1
#define result_abs r3
swi_divide:
// Set if numerator is signed, and abs numerator
ands numerator_signed, numerator, #0x80000000
rsbmi numerator, numerator, #0
// Same with denominator
ands denominator_signed, denominator, #0x80000000
rsbmi denominator, denominator, #0
// Gets set if sign(numerator) != sign(denominator)
eor sign_flip, numerator_signed, denominator_signed
mov accumulator, #0
mov current_bit, #1
// This moves out the current bit to the MSB of the denominator,
// and aligns the denominator up to the same bit-length as the
// numerator
0:
cmp denominator, numerator
movls denominator, denominator, lsl #1
movls current_bit, current_bit, lsl #1
bls 0b
// Basically the grade-school algorithm, for unsigned integers in binary
1:
cmp numerator, denominator
subcs numerator, numerator, denominator
orrcs accumulator, accumulator, current_bit
movs current_bit, current_bit, lsr #1
movcc denominator, denominator, lsr #1
bcc 1b
mov remainder, numerator
mov result_abs, accumulator
mov result, accumulator
tst sign_flip, #0x80000000
rsbmi result, result, #0
b swi_complete
#define source r0
#define dest r1
#define copy_control r2
#define value r3
#define length r12
swi_cpu_set:
// Only take 21 bits for length.
bic length, copy_control, #0xFF000000
bic length, length, #0x00E00000
tst copy_control, #(1 << 26)
bne copy_32bit
copy_16bit:
tst copy_control, #(1 << 24)
bic source, #0x1
bic dest, #0x1
bne set_16bit
0:
ldrh value, [ source ], #2
subs length, length, #1
strh value, [ dest ], #2
bne 0b
b swi_complete
set_16bit:
ldrh value, [ source ]
0:
strh value, [ dest ], #2
subs length, length, #1
bne 0b
b swi_complete
copy_32bit:
tst copy_control, #(1 << 24)
bic source, #0x3
bic dest, #0x3
bne set_32bit
0:
ldr value, [ source ], #4
subs length, length, #1
str value, [ dest ], #4
bne 0b
b swi_complete
set_32bit:
ldr value, [ source ]
0:
str value, [ dest ], #4
subs length, length, #1
bne 0b
b swi_complete
// TODO: Make this actually faster (ldm/stm)
swi_cpu_fast_set:
// Only take 21 bits for length.
bic length, copy_control, #0xFF000000
bic length, length, #0x00E00000
tst copy_control, #(1 << 24)
bic source, #0x3
bic dest, #0x3
bne fast_set_32bit
0:
ldr value, [ source ], #4
subs length, length, #1
str value, [ dest ], #4
bne 0b
b swi_complete
fast_set_32bit:
ldr value, [ source ]
0:
str value, [ dest ], #4
subs length, length, #1
bne 0b
b swi_complete
#undef remainder
#define sqrt_value r0
#define remainder r0
#define result r0
#define square_index r1
#define root r2
#define root_check r3
swi_sqrt:
mov square_index, #0x40000000
mov root, #0x0
0:
orr root_check, square_index, root
cmp remainder, root_check
subge remainder, remainder, root_check
mov root, root, lsr #1
orrge root, root, square_index
movs square_index, square_index, lsr #2
bne 0b
mov result, root
b swi_complete
#undef current_value
#define crc_value r0
#define crc_ptr r1
#define crc_length r2
#define current_value r3
#define const_0x1E r4
#define crc_index r5
#define crc_table_ptr r14
#define crc_lookup r12
#define crc_nibble(shift) ;\
and crc_index, const_0x1E, crc_value, lsl #1 ;\
ldrh crc_lookup, [ crc_table_ptr, crc_index ] ;\
mov crc_value, crc_value, lsr #4 ;\
eor crc_value, crc_value, crc_lookup ;\
;\
and crc_index, const_0x1E, current_value, shift ;\
ldrh crc_lookup, [ crc_table_ptr, crc_index ] ;\
eor crc_value, crc_value, crc_lookup ;\
crc_table:
.hword 0x0000, 0xCC01, 0xD801, 0x1400, 0xF001, 0x3C00, 0x2800, 0xE401
.hword 0xA001, 0x6C00, 0x7800, 0xB401, 0x5000, 0x9C01, 0x8801, 0x4400
swi_get_crc16:
stmdb sp!, { r5 }
mov const_0x1E, #0x1E
adr crc_table_ptr, crc_table
bic crc_value, crc_value, #0xFF000000
bic crc_value, crc_value, #0x00FF0000
movs crc_length, crc_length, lsr #1
beq 1f
0:
ldrh current_value, [ crc_ptr ], #2
crc_nibble(lsl #1)
crc_nibble(lsr #3)
crc_nibble(lsr #7)
crc_nibble(lsr #11)
subs crc_length, crc_length, #1
bne 0b
1:
ldmia sp!, { r5 }
b swi_complete
swi_is_debugger:
mov r0, #0
b swi_complete
#undef length
#define source r0
#define dest r1
#define unpack_data r2
#define length r3
#define source_bit_width r12
#define dest_bit_width r14
#define data_offset r4
#define offset_zero r2
#define source_mask r5
#define source_bit_buffer r6
#define dest_bit_buffer r7
#define dest_bits_loaded r8
#define source_value r9
swi_bit_unpack:
stmdb sp!, { r5 - r9 }
ldrh length, [ unpack_data, #0 ]
ldrb source_bit_width, [ unpack_data, #2 ]
ldrb dest_bit_width, [ unpack_data, #3 ]
ldr data_offset, [ unpack_data, #4 ]
// TODO: Check that bits are <= 32 and a power of 2?
mov offset_zero, data_offset, lsr #31
bic data_offset, data_offset, #0x80000000
mov source_mask, #0x1
mov source_mask, source_mask, lsl source_bit_width
sub source_mask, source_mask, #1
mov source_bit_buffer, #0x1
mov dest_bit_buffer, #0
mov dest_bits_loaded, #0
cmp source_bit_width, #1
moveq length, length, lsl #3
cmp source_bit_width, #2
moveq length, length, lsl #2
cmp source_bit_width, #4
moveq length, length, lsl #1
0:
// Reload bit buffer if necessary
cmp source_bit_buffer, #0x1
ldreqb source_bit_buffer, [ source ], #1
orreq source_bit_buffer, source_bit_buffer, #0x100
// Get next from source
and source_value, source_bit_buffer, source_mask
mov source_bit_buffer, source_bit_buffer, lsr source_bit_width
cmp source_value, #0
tsteq offset_zero, #0x1
addne source_value, source_value, data_offset
// Put it on dest buffer
orr dest_bit_buffer, dest_bit_buffer, source_value, lsl dest_bits_loaded
add dest_bits_loaded, dest_bits_loaded, dest_bit_width
// If dest buffer is full write out
cmp dest_bits_loaded, #32
streq dest_bit_buffer, [ dest ], #4
moveq dest_bit_buffer, #0
moveq dest_bits_loaded, #0
subs length, length, #1
bne 0b
ldmia sp!, { r5 - r9 }
b swi_complete
#undef length
#define source r0
#define dest r1
#define length r2
#define header r3
#define lz77_control r3
#define lz77_value r4
#define lz77_vram_tmp r5
#define lz77_value_b r14
#define window_length r14
#define window_offset r12
#define window_ptr r12
swi_lz77_decompress_wram:
ldr header, [ source ], #4
movs length, header, lsr #8
beq swi_complete
0:
ldrb lz77_control, [ source ], #1
// This will hit the MSB after 8 iterations
orr lz77_control, lz77_control, #(1 << 23)
1:
tst lz77_control, #0x80
beq 2f
// Load a window of data that was loaded before
// Read 16-bit unaligned value
ldrb lz77_value, [ source ], #1
ldrb lz77_value_b, [ source ], #1
orr lz77_value, lz77_value_b, lz77_value, lsl #8
// Get length and offset from lz77 value
mov window_length, lz77_value, lsr #12
bic window_offset, lz77_value, #0xF000
add window_length, window_length, #3
sub window_ptr, dest, window_offset
sub window_ptr, window_ptr, #1
3:
ldrb lz77_value, [ window_ptr ], #1
subs length, length, #1
strb lz77_value, [ dest ], #1
beq swi_complete
subs window_length, window_length, #1
bne 3b
movs lz77_control, lz77_control, lsl #1
bpl 1b
b 0b
// Load a single value
2:
ldrb lz77_value, [ source ], #1
subs length, length, #1
strb lz77_value, [ dest ], #1
beq swi_complete
movs lz77_control, lz77_control, lsl #1
bpl 1b
b 0b
swi_lz77_decompress_vram:
ldr header, [ source ], #4
movs length, header, lsr #8
beq swi_complete
stmdb sp!, { r5 }
mov r5, #0
0:
ldrb lz77_control, [ source ], #1
// This will hit the MSB after 8 iterations
orr lz77_control, lz77_control, #(1 << 23)
1:
tst lz77_control, #0x80
beq 2f
// Load a window of data that was loaded before
// Read 16-bit unaligned value
ldrb lz77_value, [ source ], #1
ldrb lz77_value_b, [ source ], #1
orr lz77_value, lz77_value_b, lz77_value, lsl #8
// Get length and offset from lz77 value
mov window_length, lz77_value, lsr #12
bic window_offset, lz77_value, #0xF000
add window_length, window_length, #3
sub window_ptr, dest, window_offset
sub window_ptr, window_ptr, #1
3:
tst dest, #1
ldreqb lz77_vram_tmp, [ window_ptr ], #1
ldrneb lz77_value, [ window_ptr ], #1
orrne lz77_value, lz77_vram_tmp, lz77_value, lsl #8
strneh lz77_value, [ dest, #-1 ]
add dest, dest, #1
subs length, length, #1
ldmeqia sp!, { r5 }
beq swi_complete
subs window_length, window_length, #1
bne 3b
movs lz77_control, lz77_control, lsl #1
bpl 1b
b 0b
// Load a single value
2:
tst dest, #1
ldreqb lz77_vram_tmp, [ source ], #1
ldrneb lz77_value, [ source ], #1
orrne lz77_value, lz77_vram_tmp, lz77_value, lsl #8
strneh lz77_value, [ dest, #-1 ]
add dest, dest, #1
subs length, length, #1
ldmeqia sp!, { r5 }
beq swi_complete
movs lz77_control, lz77_control, lsl #1
bpl 1b
b 0b
// TODO: Needs to be implemented
swi_huffman_decompress:
b swi_complete
#define source r0
#define dest r1
#define length r2
#define header r3
#define rle_control r3
#define run_length r3
#define rle_value r12
// TODO: Make a safe one for VRAM.
swi_runlength_decompress_vram:
swi_runlength_decompress_wram:
ldr header, [ source ], #4
mov length, header, lsr #8
0:
ldrb rle_control, [ source ], #1
tst rle_control, #0x80
and run_length, rle_control, #0x7F
beq 1f
ldrb rle_value, [ source ], #1
add run_length, run_length, #3
2:
strb rle_value, [ dest ], #1
subs length, length, #1
beq swi_complete
subs run_length, run_length, #1
bne 2b
b 0b
1:
add run_length, run_length, #1
2:
ldrb rle_value, [ source ], #1
subs length, length, #1
strb rle_value, [ dest ], #1
beq swi_complete
subs run_length, run_length, #1
bne 2b
b 0b
#ifdef BIOS_ARM9
#undef accumulator
#undef length
#define source r0
#define dest r1
#define length r2
#define header r3
#define accumulator r12
#define current_value r3
swi_diff_8bit_unfilter_wram:
ldr header, [ source ], #4
ldrb accumulator, [ source ], #1
mov length, header, lsr #8
strb accumulator, [ dest ], #1
sub length, length, #1
0:
ldrb current_value, [ source ], #1
subs length, length, #1
add accumulator, accumulator, current_value
strb accumulator, [ dest ], #1
bne 0b
b swi_complete
swi_diff_16bit_unfilter:
ldr header, [ source ], #4
ldrh accumulator, [ source ], #2
mov length, header, lsr #8
strh accumulator, [ dest ], #2
bic length, length, #0x1
sub length, length, #2
0:
ldrh current_value, [ source ], #2
subs length, length, #2
add accumulator, accumulator, current_value
strh accumulator, [ dest ], #2
bne 0b
b swi_complete
#endif
#ifdef BIOS_ARM7
sine_table:
.hword 0x0000, 0x0324, 0x0648, 0x096A, 0x0C8C, 0x0FAB, 0x12C8, 0x15E2
.hword 0x18F9, 0x1C0B, 0x1F1A, 0x2223, 0x2528, 0x2826, 0x2B1F, 0x2E11
.hword 0x30FB, 0x33DF, 0x36BA, 0x398C, 0x3C56, 0x3F17, 0x41CE, 0x447A
.hword 0x471C, 0x49B4, 0x4C3F, 0x4EBF, 0x5133, 0x539B, 0x55F5, 0x5842
.hword 0x5A82, 0x5CB3, 0x5ED7, 0x60EB, 0x62F1, 0x64E8, 0x66CF, 0x68A6
.hword 0x6A6D, 0x6C23, 0x6DC9, 0x6F5E, 0x70E2, 0x7254, 0x73B5, 0x7504
.hword 0x7641, 0x776B, 0x7884, 0x7989, 0x7A7C, 0x7B5C, 0x7C29, 0x7CE3
.hword 0x7D89, 0x7E1D, 0x7E9C, 0x7F09, 0x7F61, 0x7FA6, 0x7FD8, 0x7FF5
swi_get_sine_table:
add r0, r0, r0
adr r1, sine_table
// Should some protection be here?
ldrh r0, [ r1, r0 ]
b swi_complete
pitch_table:
.hword 0x0000, 0x003B, 0x0076, 0x00B2, 0x00ED, 0x0128, 0x0164, 0x019F
.hword 0x01DB, 0x0217, 0x0252, 0x028E, 0x02CA, 0x0305, 0x0341, 0x037D
.hword 0x03B9, 0x03F5, 0x0431, 0x046E, 0x04AA, 0x04E6, 0x0522, 0x055F
.hword 0x059B, 0x05D8, 0x0614, 0x0651, 0x068D, 0x06CA, 0x0707, 0x0743
.hword 0x0780, 0x07BD, 0x07FA, 0x0837, 0x0874, 0x08B1, 0x08EF, 0x092C
.hword 0x0969, 0x09A7, 0x09E4, 0x0A21, 0x0A5F, 0x0A9C, 0x0ADA, 0x0B18
.hword 0x0B56, 0x0B93, 0x0BD1, 0x0C0F, 0x0C4D, 0x0C8B, 0x0CC9, 0x0D07
.hword 0x0D45, 0x0D84, 0x0DC2, 0x0E00, 0x0E3F, 0x0E7D, 0x0EBC, 0x0EFA
.hword 0x0F39, 0x0F78, 0x0FB6, 0x0FF5, 0x1034, 0x1073, 0x10B2, 0x10F1
.hword 0x1130, 0x116F, 0x11AE, 0x11EE, 0x122D, 0x126C, 0x12AC, 0x12EB
.hword 0x132B, 0x136B, 0x13AA, 0x13EA, 0x142A, 0x146A, 0x14A9, 0x14E9
.hword 0x1529, 0x1569, 0x15AA, 0x15EA, 0x162A, 0x166A, 0x16AB, 0x16EB
.hword 0x172C, 0x176C, 0x17AD, 0x17ED, 0x182E, 0x186F, 0x18B0, 0x18F0
.hword 0x1931, 0x1972, 0x19B3, 0x19F5, 0x1A36, 0x1A77, 0x1AB8, 0x1AFA
.hword 0x1B3B, 0x1B7D, 0x1BBE, 0x1C00, 0x1C41, 0x1C83, 0x1CC5, 0x1D07
.hword 0x1D48, 0x1D8A, 0x1DCC, 0x1E0E, 0x1E51, 0x1E93, 0x1ED5, 0x1F17
.hword 0x1F5A, 0x1F9C, 0x1FDF, 0x2021, 0x2064, 0x20A6, 0x20E9, 0x212C
.hword 0x216F, 0x21B2, 0x21F5, 0x2238, 0x227B, 0x22BE, 0x2301, 0x2344
.hword 0x2388, 0x23CB, 0x240E, 0x2452, 0x2496, 0x24D9, 0x251D, 0x2561
.hword 0x25A4, 0x25E8, 0x262C, 0x2670, 0x26B4, 0x26F8, 0x273D, 0x2781
.hword 0x27C5, 0x280A, 0x284E, 0x2892, 0x28D7, 0x291C, 0x2960, 0x29A5
.hword 0x29EA, 0x2A2F, 0x2A74, 0x2AB9, 0x2AFE, 0x2B43, 0x2B88, 0x2BCD
.hword 0x2C13, 0x2C58, 0x2C9D, 0x2CE3, 0x2D28, 0x2D6E, 0x2DB4, 0x2DF9
.hword 0x2E3F, 0x2E85, 0x2ECB, 0x2F11, 0x2F57, 0x2F9D, 0x2FE3, 0x302A
.hword 0x3070, 0x30B6, 0x30FD, 0x3143, 0x318A, 0x31D0, 0x3217, 0x325E
.hword 0x32A5, 0x32EC, 0x3332, 0x3379, 0x33C1, 0x3408, 0x344F, 0x3496
.hword 0x34DD, 0x3525, 0x356C, 0x35B4, 0x35FB, 0x3643, 0x368B, 0x36D3
.hword 0x371A, 0x3762, 0x37AA, 0x37F2, 0x383A, 0x3883, 0x38CB, 0x3913
.hword 0x395C, 0x39A4, 0x39ED, 0x3A35, 0x3A7E, 0x3AC6, 0x3B0F, 0x3B58
.hword 0x3BA1, 0x3BEA, 0x3C33, 0x3C7C, 0x3CC5, 0x3D0E, 0x3D58, 0x3DA1
.hword 0x3DEA, 0x3E34, 0x3E7D, 0x3EC7, 0x3F11, 0x3F5A, 0x3FA4, 0x3FEE
.hword 0x4038, 0x4082, 0x40CC, 0x4116, 0x4161, 0x41AB, 0x41F5, 0x4240
.hword 0x428A, 0x42D5, 0x431F, 0x436A, 0x43B5, 0x4400, 0x444B, 0x4495
.hword 0x44E1, 0x452C, 0x4577, 0x45C2, 0x460D, 0x4659, 0x46A4, 0x46F0
.hword 0x473B, 0x4787, 0x47D3, 0x481E, 0x486A, 0x48B6, 0x4902, 0x494E
.hword 0x499A, 0x49E6, 0x4A33, 0x4A7F, 0x4ACB, 0x4B18, 0x4B64, 0x4BB1
.hword 0x4BFE, 0x4C4A, 0x4C97, 0x4CE4, 0x4D31, 0x4D7E, 0x4DCB, 0x4E18
.hword 0x4E66, 0x4EB3, 0x4F00, 0x4F4E, 0x4F9B, 0x4FE9, 0x5036, 0x5084
.hword 0x50D2, 0x5120, 0x516E, 0x51BC, 0x520A, 0x5258, 0x52A6, 0x52F4
.hword 0x5343, 0x5391, 0x53E0, 0x542E, 0x547D, 0x54CC, 0x551A, 0x5569
.hword 0x55B8, 0x5607, 0x5656, 0x56A5, 0x56F4, 0x5744, 0x5793, 0x57E2
.hword 0x5832, 0x5882, 0x58D1, 0x5921, 0x5971, 0x59C1, 0x5A10, 0x5A60
.hword 0x5AB0, 0x5B01, 0x5B51, 0x5BA1, 0x5BF1, 0x5C42, 0x5C92, 0x5CE3
.hword 0x5D34, 0x5D84, 0x5DD5, 0x5E26, 0x5E77, 0x5EC8, 0x5F19, 0x5F6A
.hword 0x5FBB, 0x600D, 0x605E, 0x60B0, 0x6101, 0x6153, 0x61A4, 0x61F6
.hword 0x6248, 0x629A, 0x62EC, 0x633E, 0x6390, 0x63E2, 0x6434, 0x6487
.hword 0x64D9, 0x652C, 0x657E, 0x65D1, 0x6624, 0x6676, 0x66C9, 0x671C
.hword 0x676F, 0x67C2, 0x6815, 0x6869, 0x68BC, 0x690F, 0x6963, 0x69B6
.hword 0x6A0A, 0x6A5E, 0x6AB1, 0x6B05, 0x6B59, 0x6BAD, 0x6C01, 0x6C55
.hword 0x6CAA, 0x6CFE, 0x6D52, 0x6DA7, 0x6DFB, 0x6E50, 0x6EA4, 0x6EF9
.hword 0x6F4E, 0x6FA3, 0x6FF8, 0x704D, 0x70A2, 0x70F7, 0x714D, 0x71A2
.hword 0x71F7, 0x724D, 0x72A2, 0x72F8, 0x734E, 0x73A4, 0x73FA, 0x7450
.hword 0x74A6, 0x74FC, 0x7552, 0x75A8, 0x75FF, 0x7655, 0x76AC, 0x7702
.hword 0x7759, 0x77B0, 0x7807, 0x785E, 0x78B4, 0x790C, 0x7963, 0x79BA
.hword 0x7A11, 0x7A69, 0x7AC0, 0x7B18, 0x7B6F, 0x7BC7, 0x7C1F, 0x7C77
.hword 0x7CCF, 0x7D27, 0x7D7F, 0x7DD7, 0x7E2F, 0x7E88, 0x7EE0, 0x7F38
.hword 0x7F91, 0x7FEA, 0x8042, 0x809B, 0x80F4, 0x814D, 0x81A6, 0x81FF
.hword 0x8259, 0x82B2, 0x830B, 0x8365, 0x83BE, 0x8418, 0x8472, 0x84CB
.hword 0x8525, 0x857F, 0x85D9, 0x8633, 0x868E, 0x86E8, 0x8742, 0x879D
.hword 0x87F7, 0x8852, 0x88AC, 0x8907, 0x8962, 0x89BD, 0x8A18, 0x8A73
.hword 0x8ACE, 0x8B2A, 0x8B85, 0x8BE0, 0x8C3C, 0x8C97, 0x8CF3, 0x8D4F
.hword 0x8DAB, 0x8E07, 0x8E63, 0x8EBF, 0x8F1B, 0x8F77, 0x8FD4, 0x9030
.hword 0x908C, 0x90E9, 0x9146, 0x91A2, 0x91FF, 0x925C, 0x92B9, 0x9316
.hword 0x9373, 0x93D1, 0x942E, 0x948C, 0x94E9, 0x9547, 0x95A4, 0x9602
.hword 0x9660, 0x96BE, 0x971C, 0x977A, 0x97D8, 0x9836, 0x9895, 0x98F3
.hword 0x9952, 0x99B0, 0x9A0F, 0x9A6E, 0x9ACD, 0x9B2C, 0x9B8B, 0x9BEA
.hword 0x9C49, 0x9CA8, 0x9D08, 0x9D67, 0x9DC7, 0x9E26, 0x9E86, 0x9EE6
.hword 0x9F46, 0x9FA6, 0xA006, 0xA066, 0xA0C6, 0xA127, 0xA187, 0xA1E8
.hword 0xA248, 0xA2A9, 0xA30A, 0xA36B, 0xA3CC, 0xA42D, 0xA48E, 0xA4EF
.hword 0xA550, 0xA5B2, 0xA613, 0xA675, 0xA6D6, 0xA738, 0xA79A, 0xA7FC
.hword 0xA85E, 0xA8C0, 0xA922, 0xA984, 0xA9E7, 0xAA49, 0xAAAC, 0xAB0E
.hword 0xAB71, 0xABD4, 0xAC37, 0xAC9A, 0xACFD, 0xAD60, 0xADC3, 0xAE27
.hword 0xAE8A, 0xAEED, 0xAF51, 0xAFB5, 0xB019, 0xB07C, 0xB0E0, 0xB145
.hword 0xB1A9, 0xB20D, 0xB271, 0xB2D6, 0xB33A, 0xB39F, 0xB403, 0xB468
.hword 0xB4CD, 0xB532, 0xB597, 0xB5FC, 0xB662, 0xB6C7, 0xB72C, 0xB792
.hword 0xB7F7, 0xB85D, 0xB8C3, 0xB929, 0xB98F, 0xB9F5, 0xBA5B, 0xBAC1
.hword 0xBB28, 0xBB8E, 0xBBF5, 0xBC5B, 0xBCC2, 0xBD29, 0xBD90, 0xBDF7
.hword 0xBE5E, 0xBEC5, 0xBF2C, 0xBF94, 0xBFFB, 0xC063, 0xC0CA, 0xC132
.hword 0xC19A, 0xC202, 0xC26A, 0xC2D2, 0xC33A, 0xC3A2, 0xC40B, 0xC473
.hword 0xC4DC, 0xC544, 0xC5AD, 0xC616, 0xC67F, 0xC6E8, 0xC751, 0xC7BB
.hword 0xC824, 0xC88D, 0xC8F7, 0xC960, 0xC9CA, 0xCA34, 0xCA9E, 0xCB08
.hword 0xCB72, 0xCBDC, 0xCC47, 0xCCB1, 0xCD1B, 0xCD86, 0xCDF1, 0xCE5B
.hword 0xCEC6, 0xCF31, 0xCF9C, 0xD008, 0xD073, 0xD0DE, 0xD14A, 0xD1B5
.hword 0xD221, 0xD28D, 0xD2F8, 0xD364, 0xD3D0, 0xD43D, 0xD4A9, 0xD515
.hword 0xD582, 0xD5EE, 0xD65B, 0xD6C7, 0xD734, 0xD7A1, 0xD80E, 0xD87B
.hword 0xD8E9, 0xD956, 0xD9C3, 0xDA31, 0xDA9E, 0xDB0C, 0xDB7A, 0xDBE8
.hword 0xDC56, 0xDCC4, 0xDD32, 0xDDA0, 0xDE0F, 0xDE7D, 0xDEEC, 0xDF5B
.hword 0xDFC9, 0xE038, 0xE0A7, 0xE116, 0xE186, 0xE1F5, 0xE264, 0xE2D4
.hword 0xE343, 0xE3B3, 0xE423, 0xE493, 0xE503, 0xE573, 0xE5E3, 0xE654
.hword 0xE6C4, 0xE735, 0xE7A5, 0xE816, 0xE887, 0xE8F8, 0xE969, 0xE9DA
.hword 0xEA4B, 0xEABC, 0xEB2E, 0xEB9F, 0xEC11, 0xEC83, 0xECF5, 0xED66
.hword 0xEDD9, 0xEE4B, 0xEEBD, 0xEF2F, 0xEFA2, 0xF014, 0xF087, 0xF0FA
.hword 0xF16D, 0xF1E0, 0xF253, 0xF2C6, 0xF339, 0xF3AD, 0xF420, 0xF494
.hword 0xF507, 0xF57B, 0xF5EF, 0xF663, 0xF6D7, 0xF74C, 0xF7C0, 0xF834
.hword 0xF8A9, 0xF91E, 0xF992, 0xFA07, 0xFA7C, 0xFAF1, 0xFB66, 0xFBDC
.hword 0xFC51, 0xFCC7, 0xFD3C, 0xFDB2, 0xFE28, 0xFE9E, 0xFF14, 0xFF8A
pitch_table_ptr:
.word pitch_table
swi_get_pitch_table:
add r0, r0, r0
ldr r1, pitch_table_ptr
// Should some protection be here?
ldrh r0, [ r1, r0 ]
b swi_complete
volume_table: