File: strstr.s

package info (click to toggle)
cc65 2.19-2
  • links: PTS
  • area: main
  • in suites: forky, sid, trixie
  • size: 20,268 kB
  • sloc: ansic: 117,151; asm: 66,339; pascal: 4,248; makefile: 1,009; perl: 607
file content (95 lines) | stat: -rw-r--r-- 2,610 bytes parent folder | download | duplicates (2)
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
;
; Ullrich von Bassewitz, 11.12.1998
;
; char* strstr (const char* haystack, const char* needle);
;

        .export         _strstr
        .import         popptr1
        .importzp       ptr1, ptr2, ptr3, ptr4, tmp1

_strstr:
        sta     ptr2            ; Save needle
        stx     ptr2+1
        sta     ptr4            ; Setup temp copy for later

        jsr     popptr1         ; Get haystack to ptr1

; If needle is empty, return haystack

        ; ldy     #$00            Y=0 guaranteed by popptr1
        lda     (ptr2),y        ; Get first byte of needle
        beq     @Found          ; Needle is empty --> we're done

; 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).

        sta     tmp1            ; Save start of needle
@L1:    lda     (ptr1),y        ; Get next char from haystack
        beq     @NotFound       ; Jump if end
        cmp     tmp1            ; Start of needle found?
        beq     @L2             ; Jump if so
        iny                     ; Next char
        bne     @L1
        inc     ptr1+1          ; Bump high byte
        bne     @L1             ; Branch always

; We found the start of needle in haystack

@L2:    tya                     ; Get offset
        clc
        adc     ptr1
        sta     ptr1            ; Make ptr1 point to start
        bcc     @L3
        inc     ptr1+1

; ptr1 points to the start of needle now. Setup temporary pointers for the
; search. The low byte of ptr4 is already set.

@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

; Do the compare

@L4:    lda     (ptr4),y        ; Get char from needle
        beq     @Found          ; Jump if end of needle (-> found)
        cmp     (ptr3),y        ; Compare with haystack
        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)

; The strings did not compare equal, search next start of needle

@L5:    ldy     #1              ; Start after this char
        bne     @L1             ; Branch always

; We found the start of needle

@Found: lda     ptr1
        ldx     ptr1+1
        rts

; We reached end of haystack without finding needle

@NotFound:
        lda     #$00            ; return NULL
        tax
        rts