2000-05-28 13:40:48 +00:00
|
|
|
;
|
|
|
|
; Ullrich von Bassewitz, 11.12.1998
|
|
|
|
;
|
|
|
|
; char* strstr (const char* haystack, const char* needle);
|
|
|
|
;
|
|
|
|
|
2024-04-19 07:57:47 +02:00
|
|
|
.export _strstr
|
|
|
|
.import popptr1
|
|
|
|
.importzp ptr1, ptr2, ptr3, ptr4, tmp1
|
2000-05-28 13:40:48 +00:00
|
|
|
|
|
|
|
_strstr:
|
2013-05-09 13:56:54 +02:00
|
|
|
sta ptr2 ; Save needle
|
|
|
|
stx ptr2+1
|
|
|
|
sta ptr4 ; Setup temp copy for later
|
2000-05-28 13:40:48 +00:00
|
|
|
|
2018-05-20 15:30:18 +02:00
|
|
|
jsr popptr1 ; Get haystack to ptr1
|
2000-05-28 13:40:48 +00:00
|
|
|
|
|
|
|
; If needle is empty, return haystack
|
|
|
|
|
2018-05-20 15:30:18 +02:00
|
|
|
; ldy #$00 Y=0 guaranteed by popptr1
|
2013-05-09 13:56:54 +02:00
|
|
|
lda (ptr2),y ; Get first byte of needle
|
|
|
|
beq @Found ; Needle is empty --> we're done
|
2000-05-28 13:40:48 +00:00
|
|
|
|
|
|
|
; Search for the beginning of the string (this is not an optimal search
|
|
|
|
; strategy [in fact, it's pretty dumb], but it's simple to implement).
|
|
|
|
|
2024-03-19 18:07:17 +01:00
|
|
|
sta tmp1 ; Save start of needle
|
2013-05-09 13:56:54 +02:00
|
|
|
@L1: lda (ptr1),y ; Get next char from haystack
|
|
|
|
beq @NotFound ; Jump if end
|
2024-03-19 18:07:17 +01:00
|
|
|
cmp tmp1 ; Start of needle found?
|
2013-05-09 13:56:54 +02:00
|
|
|
beq @L2 ; Jump if so
|
|
|
|
iny ; Next char
|
|
|
|
bne @L1
|
|
|
|
inc ptr1+1 ; Bump high byte
|
|
|
|
bne @L1 ; Branch always
|
2000-05-28 13:40:48 +00:00
|
|
|
|
|
|
|
; We found the start of needle in haystack
|
|
|
|
|
2013-05-09 13:56:54 +02:00
|
|
|
@L2: tya ; Get offset
|
|
|
|
clc
|
|
|
|
adc ptr1
|
|
|
|
sta ptr1 ; Make ptr1 point to start
|
|
|
|
bcc @L3
|
|
|
|
inc ptr1+1
|
2000-05-28 13:40:48 +00:00
|
|
|
|
2024-04-19 07:57:47 +02:00
|
|
|
; ptr1 points to the start of needle now. Setup temporary pointers for the
|
2000-05-28 13:40:48 +00:00
|
|
|
; search. The low byte of ptr4 is already set.
|
|
|
|
|
2013-05-09 13:56:54 +02:00
|
|
|
@L3: sta ptr3
|
|
|
|
lda ptr1+1
|
|
|
|
sta ptr3+1
|
|
|
|
lda ptr2+1
|
|
|
|
sta ptr4+1
|
|
|
|
ldy #1 ; First char is identical, so start on second
|
2000-05-28 13:40:48 +00:00
|
|
|
|
|
|
|
; Do the compare
|
|
|
|
|
2013-05-09 13:56:54 +02:00
|
|
|
@L4: lda (ptr4),y ; Get char from needle
|
|
|
|
beq @Found ; Jump if end of needle (-> found)
|
2024-04-19 07:57:47 +02:00
|
|
|
cmp (ptr3),y ; Compare with haystack
|
2013-05-09 13:56:54 +02:00
|
|
|
bne @L5 ; Jump if not equal
|
|
|
|
iny ; Next char
|
|
|
|
bne @L4
|
|
|
|
inc ptr3+1
|
|
|
|
inc ptr4+1 ; Bump hi byte of pointers
|
|
|
|
bne @L4 ; Next char (branch always)
|
2000-05-28 13:40:48 +00:00
|
|
|
|
|
|
|
; The strings did not compare equal, search next start of needle
|
|
|
|
|
2013-05-09 13:56:54 +02:00
|
|
|
@L5: ldy #1 ; Start after this char
|
|
|
|
bne @L1 ; Branch always
|
2000-05-28 13:40:48 +00:00
|
|
|
|
|
|
|
; We found the start of needle
|
|
|
|
|
2013-05-09 13:56:54 +02:00
|
|
|
@Found: lda ptr1
|
|
|
|
ldx ptr1+1
|
|
|
|
rts
|
2000-05-28 13:40:48 +00:00
|
|
|
|
|
|
|
; We reached end of haystack without finding needle
|
|
|
|
|
|
|
|
@NotFound:
|
2024-04-19 07:57:47 +02:00
|
|
|
lda #$00 ; return NULL
|
|
|
|
tax
|
|
|
|
rts
|