Microcode/ucode/sort3.uc
2019-04-26 22:27:21 +02:00

120 lines
2.3 KiB
Ucode

#optable 0x0 @BREAK
#optable 0x1 @BREAK
#optable 0x2 @BREAK
#optable 0x3 @BREAK
#optable 0x4 @BREAK
#optable 0x5 @BREAK
#optable 0x6 @BREAK
#optable 0x7 @BREAK
#optable 0x9 @BREAK
#optable 0xa @BREAK
#optable 0xb @BREAK
#optable 0xc @BREAK
#optable 0xd @BREAK
#optable 0xe @BREAK
#optable 0xf @BREAK
#define LIST_START 0xE0
#define HASH_MASK 0b1111
#define FIRST_BUCKET 0x10 // Index of start of first bucket
#define LAST_BUCKET 0xD3 // Index of start of last bucket
#define BUCKET_SIZE 13 // Each bucket an hold 13 elements before spilling
// LUT entries contain pointers to last element in bucket
// This allows the LUT to serve as both a jump-table and as bucket-headers
// Look at me being all resourceful and stuff, huh? Inb4 slowdowns ;))
#data 0x00 0x77
#data 0x01 0x84
#data 0x02 0x91
#data 0x03 0x9e
#data 0x04 0xab
#data 0x05 0xb8
#data 0x06 0xc5
#data 0x07 0xd2
#data 0x08 0x0f
#data 0x09 0x1c
#data 0x0a 0x29
#data 0x0b 0x36
#data 0x0c 0x43
#data 0x0d 0x50
#data 0x0e 0x5d
#data 0x0f 0x6a
const LIST_START
mov ar pc
// Two values sorted per iteration, so half the iterations obv ;)
// For consistency, we just decrement LC twice per iteration, though
// Had this course also stressed power use or other efficiency-related questions,
// we probably wouldn't decrement twice per iteration for the sole purpose of "clarity"
// But alas, Kent doesn't care, so why should we?
lcset 32
$BUCKET_SORT
bls @MERGE_INIT
mov pc asr; incpc
mov pm ar; mov pm ir // Use whatever value we're sorting to index GR at complete random
mov pc asr; incpc
mov pm hr
// Shift AR and HR
irl; mov pm gr
irl
irl
irl
and HASH_MASK // Completely hash value from HR
mov ar asr
mov pm ar
add 1
mov ar pm
mov ar asr
mov gr pm; declc // Mark element as sorted by decrementing LC
mov hr ar
and HASH_MASK // Completely hash value from AR
mov ar asr
mov pm ar
add 1
mov ar pm
mov ar asr
mov ir pm; declc // Mark element as sorted by decrementing LC
bra @BUCKET_SORT
//// ---- MERGE ---- ////
$MERGE_INIT
// HR points to LUT
// PC points to start of first bucket
// AR points to list
mov pc hr // PC is always 0 after bucketsort because FF+1 mod 100 = 0. Imagine that!
const FIRST_BUCKET
mov ar pc
const LIST_START
$MERGE
mov ar ir
mov hr asr
sub pm
mov pc gr
// If this was not the last bucket, branch back to start of merge
sub LAST_BUCKET
adn LAST_BUCKET
bnz @MERGE
$END
$BREAK
halt