

# VLF Comb Generator

E. Paschal, Hayden Gebhardt

February 9, 2026

## Contents

|          |                          |          |
|----------|--------------------------|----------|
| <b>1</b> | <b>Background</b>        | <b>2</b> |
| <b>2</b> | <b>Annotated Code</b>    | <b>3</b> |
| <b>3</b> | <b>Appendix</b>          | <b>6</b> |
| 3.1      | Assembly Notes . . . . . | 6        |
| 3.1.1    | Source Code . . . . .    | 6        |
| 3.2      | C Notes . . . . .        | 9        |
| 3.2.1    | Source Code . . . . .    | 9        |

# 1 Background

The programs below simulate an 11-stage shift register with feedback to produce a maximal-length shift register sequence. Such a sequence is a pseudo-random sequence. Its auto-correlation is 2047 for offsets of 0 and multiples of 2047, and -1 for all other offsets. The spectrum of the output of the device is a comb spectrum.

With an oscillator input of 16.384 MHz, the internal clock runs at 16.384/4 or 4.096 MHz. The cycle time for the shift register loop is 8 instructions, so the virtual clock for the shift register is 512 kHz. With a cycle length of 2047 shifts, the pseudo-random sequence repeats at a rate of 250.1221 Hz, which is the spacing of the components of the comb spectrum.



Figure 1: Illustration of expected responses for comb calibration.

The microprocessor used to execute this code is the PIC12F629. Its pin configuration can be seen here:



Figure 2: PIC12F629 pin-out

## 2 Annotated Code

Listing 1: Initial Configuration

```
1 ; Processor Inclusion
2 ;*****
3
4 LIST p12F629           ;processor
5 include <p12f629.inc> ; header file for MC.
6
7 ; Configuration Word
8 ;*****
9
10 ; CONFIG
11     __CONFIG _BODEN_ON    ; Brown out detection enabled
12     __CONFIG _MCLRE_OFF   ; Master clear unset, GP3 is GPIO
13     __CONFIG _PWRTE_OFF   ; Power-up timer disabled
14     __CONFIG _WDT_OFF     ; Watchdog timer disabled
15     __CONFIG _EC_OSC      ; Oscillator set to external clock
16                           ; External clock is 16 MHz
17
18 ; Variable definitions
19 ;*****
20
21 plusout EQU b'00010100' ; +CAL high, -CAL low
22 feedback EQU 0xEB        ; Feedback bits for 2047-long
23                                ; sequence, this particular
24                                ; feedback also sets +CAL low,
25                                ; -CAL high as GPIO
26 byteL   EQU 0x20          ; Lower bits allocated for LFSR
27 byteH   EQU 0x21          ; Upper bits allocated for LFSR
28 C       EQU b'00000000' ; Carry register starting seed
```

Listing 2: Start sequence, GPIO configuration, and beginning of loop

```
1 ; Reset Vector
2 ;*****
3     ORG      0x0
4     GOTO    Start ; Go to beginning of program
5
6 ; Driver Core
7 ;*****
8     ORG      0x010
9 Start
10    BANKSEL TRISIO      ; Set bank that contains TRISIO
11    MOVLW  b'11101000'
12    MOVWF  TRISIO      ; Set GPIO 0,1,2,4 as outputs
13
14    BANKSEL CMCON      ; Set bank that contains CMCON
15    MOVLW  b'00000111'
16    MOVWF  CMCON      ; Set GPIO 0,1,2 to digital I/O
17    CLRF   byteL       ; Clear registers to initialize
18    CLRF   byteH
19 Loop0           ; Start of infinite loop
20    MOVLW  plusout
21    MOVWF  GPIO        ; Carry 1 - high output
22    BCF    STATUS, C
23    RLF    byteL, F
24    RLF    byteH, F    ; Shift registers left
25    BTFSC byteH, 3    ; Test bit 12 (carry out)
26    GOTO   Loop0      ; Carry 1
27    MOVLW  feedback
28    XORWF byteL, F    ; Insert feedback
29    MOVWF  GPIO        ; Low output
30    BCF    STATUS, C
31    RLF    byteL, F
32    RLF    byteH, F    ; Shift registers left
33    BTFSC byteH, 3    ; Test bit 12 (carry out)
```

Listing 3: Full loop instructions and overflow catch

```

1   GOTO    Loop0          ; Carry = 1
2   MOVLW   feedback       ; 2nd carry = 0
3   XORWF   byteL, F      ; Insert feedback
4   MOVWF   GPIO           ; Low output
5   BCF     STATUS, C
6   RLF     byteL, F
7   RLF     byteH, F       ; Shift registers left
8   BTFSC  byteH, 3        ; Retest bit 12 (carry out)
9
10  ; ... [Instructions for carries 3-8] ...
11
12  GOTO    Loop0          ; Carry = 1
13  MOVLW   feedback       ; 9th carry = 0
14  XORWF   byteL, F      ; Insert feedback
15  MOVWF   GPIO           ; Low output
16  BCF     STATUS, C
17  RLF     byteL, F
18  RLF     byteH, F       ; Shift registers left
19  BTFSC  byteH, 3        ; Retest bit 12 (carry out)
20  GOTO    Loop0          ; Carry = 1
21  MOVLW   feedback       ; 10th carry = 0
22  XORWF   byteL, F      ; Insert feedback
23  MOVWF   GPIO           ; Low output
24  BCF     STATUS, C
25  RLF     byteL, F
26  RLF     byteH, F       ; Shift registers left
27  BTFSC  byteH, 3        ; Retest bit 12 (carry out)
28  GOTO    Loop0          ; Carry = 1
29
30  Loop1
31  MOVLW   feedback       ; 11th carry 0 (cannot happen)
32  MOVWF   GPIO           ; Sequence cannot exceed
33                      ; maximal length, set pins low
34  GOTO    Loop1          ; Repeat until restart
35
36  END    ;End of program

```

### 3 Appendix

#### 3.1 Assembly Notes

This code was written in MPLABX ver. 5.35 using the MPASM compiler. The configuration word and pre-processor instructions are defined specifically for the PIC12F629 microcontroller. The number of loop instructions correspond to an 11-stage linear feedback shift register, designed to produce spectral combs spaced 244.26 Hz apart.

##### 3.1.1 Source Code

```
1 ; Processor Inclusion
2 ;*****
3
4     LIST p=12F629      ; processor
5     #include <p12f629.inc>
6
7 ; Configuration Word
8 ;*****
9
10 ; CONFIG
11 ; _config 0x31F3
12     __CONFIG _BOREN_ON & _MCLRE_OFF & _PWRTE_OFF & _WDT_OFF &
13     _EC_OSC
14 ; Variable definitions
15 ;*****
16
17 plusout    EQU      b '00010100'
18 feedback   EQU      0xEB
19 byteL     EQU      0x20
20 byteH     EQU      0x21
21 C         EQU      b '00000000'
22
23 ; Reset Vector
24 ;*****
25
26     ORG      0x0
27     GOTO    Start
28
29 ; Driver Core
30 ;*****
```

```

31
32     ORG      0x010
33 Start
34     BANKSEL TRISIO
35     MOVLW   b'11101000'
36     MOVWF   TRISIO
37
38     BANKSEL CMCON
39     MOVLW   b'00000111'
40     MOVWF   CMCON
41     CLRF    byteL
42     CLRF    byteH
43
44 Loop0
45     MOVLW   plusout
46     MOVWF   GPIO
47     BCF     STATUS, C
48     RLF     byteL, F
49     RLF     byteH, F
50     BTFSC  byteH, 3
51     GOTO   Loop0
52     MOVLW   feedback
53     XORWF  byteL, F
54     MOVWF   GPIO
55     BCF     STATUS, C
56     RLF     byteL, F
57     RLF     byteH, F
58     BTFSC  byteH, 3
59     GOTO   Loop0
60     MOVLW   feedback
61     XORWF  byteL, F
62     MOVWF   GPIO
63     BCF     STATUS, C
64     RLF     byteL, F
65     RLF     byteH, F
66     BTFSC  byteH, 3
67     GOTO   Loop0
68     MOVLW   feedback
69     XORWF  byteL, F
70     MOVWF   GPIO
71     BCF     STATUS, C
72     RLF     byteL, F
73     RLF     byteH, F
74     BTFSC  byteH, 3
75     GOTO   Loop0

```

```

76    MOVLW   feedback
77    XORWF   byteL, F
78    MOVWF   GPIO
79    BCF     STATUS, C
80    RLF     byteL, F
81    RLF     byteH, F
82    BTFSC   byteH, 3
83    GOTO    Loop0
84    MOVLW   feedback
85    XORWF   byteL, F
86    MOVWF   GPIO
87    BCF     STATUS, C
88    RLF     byteL, F
89    RLF     byteH, F
90    BTFSC   byteH, 3
91    GOTO    Loop0
92    MOVLW   feedback
93    XORWF   byteL, F
94    MOVWF   GPIO
95    BCF     STATUS, C
96    RLF     byteL, F
97    RLF     byteH, F
98    BTFSC   byteH, 3
99    GOTO    Loop0
100   MOVLW   feedback
101   XORWF   byteL, F
102   MOVWF   GPIO
103   BCF     STATUS, C
104   RLF     byteL, F
105   RLF     byteH, F
106   BTFSC   byteH, 3
107   GOTO    Loop0
108   MOVLW   feedback
109   XORWF   byteL, F
110   MOVWF   GPIO
111   BCF     STATUS, C
112   RLF     byteL, F
113   RLF     byteH, F
114   BTFSC   byteH, 3
115   GOTO    Loop0
116   MOVLW   feedback
117   XORWF   byteL, F
118   MOVWF   GPIO
119   BCF     STATUS, C
120   RLF     byteL, F

```

```

121    RLF    byteH, F
122    BTFSC byteH, 3
123    GOTO   Loop0
124    MOVLW  feedback
125    XORWF byteL, F
126    MOVWF  GPIO
127    BCF    STATUS, C
128    RLF    byteL, F
129    RLF    byteH, F
130    BTFSC byteH, 3
131    GOTO   Loop0
132
133 Loop1
134    MOVLW  feedback
135    MOVWF  GPIO
136    GOTO   Loop1
137
138 END

```

## 3.2 C Notes

This code will produce the same result as the assembly code above, but the XC8 compiler is not fast enough to generate in-phase outputs on all required GPIO pins. If one tethers two identical output pins, they would see a voltage division in the output due to the delay incurred by down-compiling the C code. Use this code if you have a faster micro-controller or a more reliable C compiler. This program was last executed on MPLAB ver. 6.10 using the supported XC8 compiler.

### 3.2.1 Source Code

```

1 #include <xc.h>
2 #include <stdio.h>
3 #include <stdint.h>
4 #include <stdlib.h>
5
6 // Configuration bits
7 #pragma config FOSC = EC      // GP4 operates as I/O, CLKIN on
                                // GP5
8 #pragma config WDIE = OFF     // Watchdog timer disabled
9 #pragma config PWRTE = OFF    // Power-up timer disabled

```

```

10 #pragma config MCLRE = OFF      // MCLR pin function is digital I/
11                                     O
12 #pragma config BOREN = OFF      // Brown-out reset disabled
13 #pragma config CP = OFF        // Code protection disabled
14 #pragma config CPD = OFF       // Data code protection disabled
15
16 void main()
17 {
18     TRISIO &= 0xE8;    // set TRISIO (0, 1, 2, 4) as output
19     GPIO = 0;          // set all declared GPIO pins as outputs
20
21     const uint16_t start = 0x002; // algorithm seed (0b 0000
22                                     0000 0010)
23     const uint16_t mask = 0x07FF; // constant mask for
24     autocorrelation range (0d 2047)
25     uint16_t a = start;         // set autocorrelation equal
26                                     to starting seed
27
28     while (1) {
29
30         unsigned int newbit = ((a >> 10) ^ (a >> 1)) & 1; // generate new bit using LSFR logic with two taps at 2^10, 2^1
31                                     bits
32         a = ((a << 1) | newbit) & mask; // apply the new bit and
33                                     mask the value to the autocorrelation
34                                     range
35
36         // ternary operation to determine if sequence is
37         // complete. If so, set all GPIO outputs equal to seed
38         GPIO = a == start ? (start & 0xFF) : (a & 0xFF << 4) |
39         (~a & 0x03); // set GPIO pins based on the autocorrelation
40         value
41     }
42 }
```