2021-06-19 20:11:26 +00:00
|
|
|
/// @file
|
2021-06-19 19:26:05 +00:00
|
|
|
/// C standard library string.h
|
2021-06-19 20:28:44 +00:00
|
|
|
///
|
2021-06-19 19:26:05 +00:00
|
|
|
/// Functions to manipulate C strings and arrays.
|
2021-08-02 20:30:04 +00:00
|
|
|
/// NULL pointer
|
2020-12-21 07:57:41 +00:00
|
|
|
// Commodore 64 PRG executable file
|
|
|
|
.file [name="sieve-min.prg", type="prg", segments="Program"]
|
|
|
|
.segmentdef Program [segments="Basic, Code, Data"]
|
|
|
|
.segmentdef Basic [start=$0801]
|
|
|
|
.segmentdef Code [start=$80d]
|
|
|
|
.segmentdef Data [startAfter="Code"]
|
|
|
|
.segment Basic
|
2019-08-07 09:35:15 +00:00
|
|
|
:BasicUpstart(main)
|
|
|
|
.const COUNT = $4000
|
|
|
|
/* Up to what number? */
|
|
|
|
.const SQRT_COUNT = $80
|
2020-04-30 20:15:59 +00:00
|
|
|
.label SCREEN = $400
|
2019-08-07 09:35:15 +00:00
|
|
|
/* Sqrt of COUNT */
|
|
|
|
.label sieve = $1000
|
2020-06-27 18:32:09 +00:00
|
|
|
.label print_screen = $400
|
2021-09-23 06:24:56 +00:00
|
|
|
.label print_char_cursor = 2
|
2020-12-21 07:57:41 +00:00
|
|
|
.segment Code
|
2019-08-07 09:35:15 +00:00
|
|
|
main: {
|
2021-09-23 06:24:56 +00:00
|
|
|
.label i = 2
|
|
|
|
.label sieve_i = $c
|
2020-03-08 22:26:49 +00:00
|
|
|
.label j = 6
|
|
|
|
.label s = 8
|
2021-09-23 06:24:56 +00:00
|
|
|
.label i_1 = $a
|
|
|
|
.label __16 = 4
|
2020-02-23 08:44:36 +00:00
|
|
|
// memset(sieve, 0, COUNT)
|
2020-03-07 21:38:40 +00:00
|
|
|
// Fill sieve with zeros
|
2019-08-07 09:35:15 +00:00
|
|
|
jsr memset
|
|
|
|
lda #<sieve+2
|
2019-08-07 19:00:19 +00:00
|
|
|
sta.z sieve_i
|
2019-08-07 09:35:15 +00:00
|
|
|
lda #>sieve+2
|
2019-08-07 19:00:19 +00:00
|
|
|
sta.z sieve_i+1
|
2019-08-07 09:35:15 +00:00
|
|
|
lda #<2
|
2019-08-25 20:21:56 +00:00
|
|
|
sta.z i
|
2019-08-07 09:35:15 +00:00
|
|
|
lda #>2
|
2019-08-25 20:21:56 +00:00
|
|
|
sta.z i+1
|
2019-09-29 21:13:37 +00:00
|
|
|
__b1:
|
2020-02-23 08:44:36 +00:00
|
|
|
// while (i < SQRT_COUNT)
|
2019-08-25 20:21:56 +00:00
|
|
|
lda.z i+1
|
2019-08-07 11:22:05 +00:00
|
|
|
bne !+
|
2019-08-25 20:21:56 +00:00
|
|
|
lda.z i
|
2020-09-28 23:35:08 +00:00
|
|
|
cmp #SQRT_COUNT
|
2019-09-29 21:13:37 +00:00
|
|
|
bcc __b2
|
2019-08-07 11:22:05 +00:00
|
|
|
!:
|
2020-06-27 18:32:09 +00:00
|
|
|
lda #<print_screen
|
2019-08-07 19:00:19 +00:00
|
|
|
sta.z print_char_cursor
|
2020-06-27 18:32:09 +00:00
|
|
|
lda #>print_screen
|
2019-08-07 19:00:19 +00:00
|
|
|
sta.z print_char_cursor+1
|
2019-08-07 11:22:05 +00:00
|
|
|
lda #<2
|
2019-10-19 23:43:51 +00:00
|
|
|
sta.z i_1
|
2019-08-07 11:22:05 +00:00
|
|
|
lda #>2
|
2019-10-19 23:43:51 +00:00
|
|
|
sta.z i_1+1
|
2019-09-29 21:13:37 +00:00
|
|
|
__b7:
|
2020-02-23 08:44:36 +00:00
|
|
|
// for (i = 2; i < 0x04c7; ++i)
|
2019-10-19 23:43:51 +00:00
|
|
|
lda.z i_1+1
|
2019-08-25 20:21:56 +00:00
|
|
|
cmp #>$4c7
|
2019-09-29 21:13:37 +00:00
|
|
|
bcc __b8
|
2019-08-25 20:21:56 +00:00
|
|
|
bne !+
|
2019-10-19 23:43:51 +00:00
|
|
|
lda.z i_1
|
2019-08-25 20:21:56 +00:00
|
|
|
cmp #<$4c7
|
2019-09-29 21:13:37 +00:00
|
|
|
bcc __b8
|
2019-08-25 20:21:56 +00:00
|
|
|
!:
|
2019-09-29 21:13:37 +00:00
|
|
|
__b11:
|
2020-02-23 08:44:36 +00:00
|
|
|
// (*(SCREEN+999))++;
|
2019-08-25 20:21:56 +00:00
|
|
|
inc SCREEN+$3e7
|
2019-09-29 21:13:37 +00:00
|
|
|
jmp __b11
|
|
|
|
__b8:
|
2020-02-23 08:44:36 +00:00
|
|
|
// if (!sieve[i])
|
2020-10-12 18:28:00 +00:00
|
|
|
lda.z i_1
|
2021-06-29 22:58:50 +00:00
|
|
|
clc
|
2019-08-07 11:22:05 +00:00
|
|
|
adc #<sieve
|
2019-11-03 16:05:55 +00:00
|
|
|
sta.z __16
|
2019-10-19 23:43:51 +00:00
|
|
|
lda.z i_1+1
|
2019-08-07 11:22:05 +00:00
|
|
|
adc #>sieve
|
2019-11-03 16:05:55 +00:00
|
|
|
sta.z __16+1
|
2019-08-07 11:22:05 +00:00
|
|
|
ldy #0
|
2019-11-03 16:05:55 +00:00
|
|
|
lda (__16),y
|
2019-08-07 11:22:05 +00:00
|
|
|
cmp #0
|
2019-09-29 21:13:37 +00:00
|
|
|
bne __b9
|
2020-04-13 18:00:13 +00:00
|
|
|
// print_uint(i)
|
|
|
|
jsr print_uint
|
2020-02-23 08:44:36 +00:00
|
|
|
// print_char(' ')
|
2019-08-07 11:22:05 +00:00
|
|
|
lda #' '
|
|
|
|
jsr print_char
|
2019-09-29 21:13:37 +00:00
|
|
|
__b9:
|
2020-02-23 08:44:36 +00:00
|
|
|
// for (i = 2; i < 0x04c7; ++i)
|
2019-10-19 23:43:51 +00:00
|
|
|
inc.z i_1
|
2019-08-25 20:21:56 +00:00
|
|
|
bne !+
|
2019-10-19 23:43:51 +00:00
|
|
|
inc.z i_1+1
|
2019-08-25 20:21:56 +00:00
|
|
|
!:
|
2019-09-29 21:13:37 +00:00
|
|
|
jmp __b7
|
|
|
|
__b2:
|
2020-02-23 08:44:36 +00:00
|
|
|
// if (!*sieve_i)
|
2019-08-25 20:21:56 +00:00
|
|
|
ldy #0
|
|
|
|
lda (sieve_i),y
|
|
|
|
cmp #0
|
2019-09-29 21:13:37 +00:00
|
|
|
bne __b3
|
2021-05-02 12:28:03 +00:00
|
|
|
// unsigned int j = i*2
|
2021-07-27 17:11:14 +00:00
|
|
|
/* Prime number - mark all multiples */
|
2019-08-25 20:21:56 +00:00
|
|
|
lda.z i
|
|
|
|
asl
|
|
|
|
sta.z j
|
|
|
|
lda.z i+1
|
|
|
|
rol
|
|
|
|
sta.z j+1
|
2021-05-02 12:28:03 +00:00
|
|
|
// unsigned char* s = &sieve[j]
|
2020-10-12 18:28:00 +00:00
|
|
|
lda.z j
|
2021-06-29 22:58:50 +00:00
|
|
|
clc
|
2019-08-25 20:21:56 +00:00
|
|
|
adc #<sieve
|
|
|
|
sta.z s
|
|
|
|
lda.z j+1
|
|
|
|
adc #>sieve
|
|
|
|
sta.z s+1
|
2019-09-29 21:13:37 +00:00
|
|
|
__b4:
|
2020-02-23 08:44:36 +00:00
|
|
|
// while (j < COUNT)
|
2019-08-25 20:21:56 +00:00
|
|
|
lda.z j+1
|
|
|
|
cmp #>COUNT
|
2019-09-29 21:13:37 +00:00
|
|
|
bcc __b5
|
2019-08-25 20:21:56 +00:00
|
|
|
bne !+
|
|
|
|
lda.z j
|
|
|
|
cmp #<COUNT
|
2019-09-29 21:13:37 +00:00
|
|
|
bcc __b5
|
2019-08-25 20:21:56 +00:00
|
|
|
!:
|
2019-09-29 21:13:37 +00:00
|
|
|
__b3:
|
2020-02-23 08:44:36 +00:00
|
|
|
// i++;
|
2019-08-07 19:00:19 +00:00
|
|
|
inc.z i
|
2019-08-07 11:22:05 +00:00
|
|
|
bne !+
|
2019-08-07 19:00:19 +00:00
|
|
|
inc.z i+1
|
2019-08-07 11:22:05 +00:00
|
|
|
!:
|
2020-02-23 08:44:36 +00:00
|
|
|
// sieve_i++;
|
2019-08-25 20:21:56 +00:00
|
|
|
inc.z sieve_i
|
2019-08-07 11:22:05 +00:00
|
|
|
bne !+
|
2019-08-25 20:21:56 +00:00
|
|
|
inc.z sieve_i+1
|
2019-08-07 11:22:05 +00:00
|
|
|
!:
|
2019-09-29 21:13:37 +00:00
|
|
|
jmp __b1
|
|
|
|
__b5:
|
2020-02-23 08:44:36 +00:00
|
|
|
// *s = 1
|
2019-08-07 09:35:15 +00:00
|
|
|
lda #1
|
|
|
|
ldy #0
|
|
|
|
sta (s),y
|
2020-02-23 08:44:36 +00:00
|
|
|
// s += i
|
2019-08-07 09:35:15 +00:00
|
|
|
clc
|
2021-06-29 22:05:01 +00:00
|
|
|
lda.z s
|
2019-08-25 20:21:56 +00:00
|
|
|
adc.z i
|
2019-08-07 19:00:19 +00:00
|
|
|
sta.z s
|
|
|
|
lda.z s+1
|
2019-08-25 20:21:56 +00:00
|
|
|
adc.z i+1
|
2019-08-07 19:00:19 +00:00
|
|
|
sta.z s+1
|
2020-02-23 08:44:36 +00:00
|
|
|
// j += i
|
2019-08-07 09:35:15 +00:00
|
|
|
clc
|
2021-06-29 22:05:01 +00:00
|
|
|
lda.z j
|
2019-08-25 20:21:56 +00:00
|
|
|
adc.z i
|
2019-08-07 19:00:19 +00:00
|
|
|
sta.z j
|
|
|
|
lda.z j+1
|
2019-08-25 20:21:56 +00:00
|
|
|
adc.z i+1
|
2019-08-07 19:00:19 +00:00
|
|
|
sta.z j+1
|
2019-09-29 21:13:37 +00:00
|
|
|
jmp __b4
|
2019-08-07 09:35:15 +00:00
|
|
|
}
|
2020-06-27 21:26:57 +00:00
|
|
|
// Copies the character c (an unsigned char) to the first num characters of the object pointed to by the argument str.
|
2021-08-10 15:48:55 +00:00
|
|
|
// void * memset(void *str, char c, unsigned int num)
|
2020-06-27 21:26:57 +00:00
|
|
|
memset: {
|
|
|
|
.const c = 0
|
|
|
|
.label str = sieve
|
|
|
|
.label end = str+COUNT
|
2021-09-23 06:24:56 +00:00
|
|
|
.label dst = 4
|
2020-06-27 21:26:57 +00:00
|
|
|
lda #<str
|
|
|
|
sta.z dst
|
|
|
|
lda #>str
|
|
|
|
sta.z dst+1
|
|
|
|
__b1:
|
|
|
|
// for(char* dst = str; dst!=end; dst++)
|
|
|
|
lda.z dst+1
|
|
|
|
cmp #>end
|
|
|
|
bne __b2
|
|
|
|
lda.z dst
|
|
|
|
cmp #<end
|
|
|
|
bne __b2
|
|
|
|
// }
|
|
|
|
rts
|
|
|
|
__b2:
|
|
|
|
// *dst = c
|
|
|
|
lda #c
|
2019-08-07 09:35:15 +00:00
|
|
|
ldy #0
|
2020-06-27 21:26:57 +00:00
|
|
|
sta (dst),y
|
|
|
|
// for(char* dst = str; dst!=end; dst++)
|
|
|
|
inc.z dst
|
2019-08-07 09:35:15 +00:00
|
|
|
bne !+
|
2020-06-27 21:26:57 +00:00
|
|
|
inc.z dst+1
|
2019-08-07 09:35:15 +00:00
|
|
|
!:
|
2020-06-27 21:26:57 +00:00
|
|
|
jmp __b1
|
2019-08-07 09:35:15 +00:00
|
|
|
}
|
2020-04-13 18:00:13 +00:00
|
|
|
// Print a unsigned int as HEX
|
2021-09-23 06:24:56 +00:00
|
|
|
// void print_uint(__zp($a) unsigned int w)
|
2020-04-13 18:00:13 +00:00
|
|
|
print_uint: {
|
2021-09-23 06:24:56 +00:00
|
|
|
.label w = $a
|
2021-06-12 18:57:35 +00:00
|
|
|
// print_uchar(BYTE1(w))
|
2020-03-22 21:26:39 +00:00
|
|
|
ldx.z w+1
|
2020-04-13 18:06:30 +00:00
|
|
|
jsr print_uchar
|
2021-06-12 18:57:35 +00:00
|
|
|
// print_uchar(BYTE0(w))
|
2020-03-22 21:26:39 +00:00
|
|
|
ldx.z w
|
2020-04-13 18:06:30 +00:00
|
|
|
jsr print_uchar
|
2020-02-23 08:44:36 +00:00
|
|
|
// }
|
2019-08-07 09:35:15 +00:00
|
|
|
rts
|
|
|
|
}
|
2020-06-27 21:26:57 +00:00
|
|
|
// Print a single char
|
2021-08-10 15:48:55 +00:00
|
|
|
// void print_char(__register(A) char ch)
|
2020-06-27 21:26:57 +00:00
|
|
|
print_char: {
|
|
|
|
// *(print_char_cursor++) = ch
|
|
|
|
ldy #0
|
|
|
|
sta (print_char_cursor),y
|
|
|
|
// *(print_char_cursor++) = ch;
|
|
|
|
inc.z print_char_cursor
|
|
|
|
bne !+
|
|
|
|
inc.z print_char_cursor+1
|
|
|
|
!:
|
|
|
|
// }
|
|
|
|
rts
|
|
|
|
}
|
2020-04-13 18:00:13 +00:00
|
|
|
// Print a char as HEX
|
2021-08-10 15:48:55 +00:00
|
|
|
// void print_uchar(__register(X) char b)
|
2020-04-13 18:06:30 +00:00
|
|
|
print_uchar: {
|
2020-02-23 08:44:36 +00:00
|
|
|
// b>>4
|
2019-08-07 09:35:15 +00:00
|
|
|
txa
|
|
|
|
lsr
|
|
|
|
lsr
|
|
|
|
lsr
|
|
|
|
lsr
|
2020-02-23 08:44:36 +00:00
|
|
|
// print_char(print_hextab[b>>4])
|
2019-08-07 09:35:15 +00:00
|
|
|
tay
|
|
|
|
lda print_hextab,y
|
2020-03-07 21:38:40 +00:00
|
|
|
// Table of hexadecimal digits
|
2019-08-07 09:35:15 +00:00
|
|
|
jsr print_char
|
2024-01-02 18:45:16 +00:00
|
|
|
// b&0xf
|
2019-08-07 09:35:15 +00:00
|
|
|
lda #$f
|
|
|
|
axs #0
|
2024-01-02 18:45:16 +00:00
|
|
|
// print_char(print_hextab[b&0xf])
|
2019-08-07 09:35:15 +00:00
|
|
|
lda print_hextab,x
|
|
|
|
jsr print_char
|
2020-02-23 08:44:36 +00:00
|
|
|
// }
|
2019-08-07 09:35:15 +00:00
|
|
|
rts
|
|
|
|
}
|
2020-12-21 07:57:41 +00:00
|
|
|
.segment Data
|
2019-08-07 09:35:15 +00:00
|
|
|
print_hextab: .text "0123456789abcdef"
|