Crystal substitute using si5351 – part 2

Crystal substitute using si5351 – part 1 described the first part of a series on an inexpensive crystal replacement using a si5351-A / MS5351M PLL chip and an ATTiny controller.

Above is an example pair of inexpensive modules, less than $10 for the pair (incl shipping). Both boards are powered from 5V, the left hand module is a ATTiny85 dev board, it has a small 3.3V regulator on board. The dev board uses a DIP chip, so it can easily be programmed in a device programmer and then inserted in the socket.

Above is a cold start of the module, error settles around +137ppm.

Above is an Allan Deviation chart for this module referenced to a Trimble Thunderbolt (module freq=10M-137.5ppm). Short term stability is not too bad, the bigger problem is likely temperature response over hours and days. The crystal on this cheap module has 140ppm error, so specification accuracy would be worse than that, and temperature stability is probably about the same for 10-60°. So, that is quite likely around 3ppm/°.

Note that the Adafruit module that this appears to be cloned from is specified with a crystal of 30ppm accuracy and similar temperature stability over 10-60° (0.6ppm/°).

Development programming

 

Above is a screenshot of the EEPROM configuration data with two selectable configuration sets (burstsets) programmed. The ATTinyx5 has two pins for selection of one of four burstsets, the ATTinyx4 chips have four pins available for selection one of up to 16 burstsets, limited in both cases by the EEPROM available.

Bus Pirate v5

A Python script was written to parse the exported configuration from Clockbuilder Pro and create the Bus Pirate commands to be pasted into a terminal emulator (Teraterm v5). Bus Pirate 5 is not backwards compatible with v4 and v3, so the script accepts an argument to set the language version. The script also writes a binary file of the EEPROM image for use with a device programmer.

Teraterm was used as it allow specification of a pause at the end of each line to allow Bus Pirate to execute the command.

#!/usr/bin/env python
# coding: utf-8

from pathlib import Path
import csv
import os
import sys
import struct

print('\n'+Path(__file__).stem+' v1.03 20240330 Owen Duffy\n')

import getopt,sys

def usage():
  print(Path(__file__).stem+" [-a hexaddress] [-m mversion][-o outprefix][-b bfilename][-p pfilename][-t toclen] ifile1 [ifilen]...")
  sys.exit(2)

try:
  opts,args=getopt.getopt(sys.argv[1:],"ha:b:m:o:p:t:v:",["help","address=","bfilenane=","mversion=","outprefix=","pfilename=","toclen=","verbose="])
except getopt.GetoptError as err:
  # print help information and exit:
  print(err)  # will print something like "option -a not recognized"
  usage()
  sys.exit(2)

verbose=False

ifilesn=0
ifilenames=[]
if len(args)<1:
  usage()
for arg in args:
  ifilenames.append(arg)
  print(arg,ifilesn)
  ifilesn=ifilesn+1
chip=0xc0
toclen=ifilesn+3
bpv=5

p=Path(ifilenames[0])
pfilename=p.with_suffix('.pllldri')
bfilename=p.with_suffix('.bp')

for o, a in opts:
  if o == "-v":
    verbose = True
  elif o in ("-h", "--help"):
    usage()
    sys.exit()
  elif o in ("-a", "--address"):
    chip=int(a,16)
  elif o in ("-m", "--mversion"):
    bpv=int(a)
  elif o in ("-b", "--bfilename"):
    bfilename=a
  elif o in ("-o", "--outprefix"):
    prefix=Path(a)
    pfilename=prefix.with_suffix('.pllldri')
    bfilename=prefix.with_suffix('.bp')
  elif o in ("-p", "--pfilename"):
    pfilename=a
  elif o in ("-t", "--toclen"):
    toclen=a
    if toclen0):
          if row[0].startswith("#"):
            continue
          addr=int(row[0])
          data=int(row[1].rstrip('h'),16)
          if(addr-1==lastaddr):
            bfile.write(' 0x{:02x}'.format(data))
            burst=burst+struct.pack('B',data)
            burstlen=burstlen+1
          else:
            bfile.write('\n[ 0x{:02x} 0x{:02x} 0x{:02x}'.format(chip,addr,data))
            if(burstlen>0):
              burst=struct.pack('B',burstlen)+burst #prepend burstlen
              pbuf[i]=pbuf[i]+burst #write to file buffer
              burstsetlen=burstsetlen+burstlen+1
              burstn=burstn+1
            burst=struct.pack('BB',addr,data)
            burstlen=2
            #burstn=burstn+1
          lastaddr=addr
  #      print(row)
  #      print('burstlen2: ',burstlen,burstn)
    if(burstlen>0):
      burst=struct.pack('B',burstlen)+burst #prepend burstlen
      pbuf[i]=pbuf[i]+burst #write to file buffer
      burstsetlen=burstsetlen+burstlen+1
      burstn=burstn+1
      pbuf[i]=struct.pack('BB',burstn,burstsetlen)+pbuf[i] #prepend burst list hdr
  #    print('burstlen2: ',burstlen,burstn)
  bfile.write(' ]\np\nw\ni\n\n')

  i=i+1

#write pfile header
buf=b'\x01\x03\x00'+struct.pack('B',chip)+struct.pack('B',len(pbuf))
#append toc
offs=5+toclen*2
i=0
for x in pbuf:
  buf=buf+struct.pack('<H',offs) #append toc entry
  offs=offs+len(x)
  i=i+1
for j in range(i,toclen):
  buf=buf+b'\xff\xff' #append empty toc records
#append burst sets
for x in pbuf:
  buf=buf+x
pfile.write(buf)
pfile.close
bfile.close()

Above is the Python script.

HiZ> m

Mode selection
 1. HiZ
 2. 1-WIRE
 3. UART
 4. HDUART
 5. I2C
 6. SPI
 7. 2WIRE
 8. DIO
 9. LED
 10. INFRARED
 x. Exit
Mode > 5

Use previous settings?
 I2C speed: 100 kHz
 Clock stretching: OFF

y/n, x to exit (Y) > n

I2C speed
 1kHz to 1000kHz
 x. Exit
kHz (400kHz*) > 100
Clock stretching
 1. OFF*
 2. ON
 x. Exit
OFF (1) > 1
Mode: I2C
I2C> W
Power supply
Volts (0.80V-5.00V)
x to exit (3.30) > 3.3
Maximum current (0mA-500mA),  for none
x to exit (none) > 100
3.30V requested, closest value: 3.30V
100.0mA requested, closest value: 100.0mA

Power supply:Enabled
Vreg output: 3.3V, Vref/Vout pin: 3.3V, Current: 28.4mA

I2C> P
Pull-up resistors: Enabled (10K ohms @ 3.3V)
I2C> i

This device complies with part 15 of the FCC Rules. Operation is subject to the following two conditions: (1) this device may not cause harmful interference, and (2) this device must accept any interference received, including interference that may cause undesired operation.


Bus Pirate 6
https://BusPirate.com/
Firmware main branch @ 8e51eb6 (2024-10-31T11:09:12Z)
RP2350B with 512KB RAM, 128Mbit FLASH
S/N: 2EAE869F8943242C
Storage:   0.10GB (FAT16 File System)

Configuration file: Loaded
Available modes: HiZ 1-WIRE UART HDUART I2C SPI 2WIRE DIO LED INFRARED
Active mode: I2C
 I2C speed: 100 kHz
 Clock stretching: OFF

Display format: Auto
Data format: 8 bits, MSB bitorder
Pull-up resistors: ON
Power supply: ON (3.3V/3.29V)
Current limit: OK (36.9mA/100.0mA)
Frequency generators: OFF

1.Vout  2.IO0   3.IO1   4.IO2   5.IO3   6.IO4   7.IO5   8.IO6   9.IO7   10.GND
36.9mA  SDA     SCL     -       -       -       -       -       -       GND
3.3V    3.3V    3.2V    3.2V    3.3V    3.3V    3.2V    3.2V    3.2V    GND

I2C> [ 0xc0 0x02 0x53 0x00 0x20

I2C START
TX: 0xC0 ACK 0x02 ACK 0x53 ACK 0x00 ACK 0x20 ACK
I2C> [ 0xc0 0x07 0x00

I2C REPEATED START
TX: 0xC0 ACK 0x07 ACK 0x00 ACK
I2C> [ 0xc0 0x0f 0x00 0x0c 0x8c 0x8c 0x8c 0x8c 0x8c 0x8c 0x8c

I2C REPEATED START
TX: 0xC0 ACK 0x0F ACK 0x00 ACK 0x0C ACK 0x8C ACK 0x8C ACK 0x8C ACK 0x8C ACK
    0x8C ACK 0x8C ACK 0x8C ACK
I2C> [ 0xc0 0x1a 0xb5 0xa7 0x00 0x0f 0xf3 0x00 0x24 0xfb

I2C REPEATED START
TX: 0xC0 ACK 0x1A ACK 0xB5 ACK 0xA7 ACK 0x00 ACK 0x0F ACK 0xF3 ACK 0x00 ACK
    0x24 ACK 0xFB ACK
I2C> [ 0xc0 0x2a 0x00 0x04 0x00 0x2a 0xe0 0x00 0x00 0x00

I2C REPEATED START
TX: 0xC0 ACK 0x2A ACK 0x00 ACK 0x04 ACK 0x00 ACK 0x2A ACK 0xE0 ACK 0x00 ACK
    0x00 ACK 0x00 ACK
I2C> [ 0xc0 0x5a 0x00 0x00

I2C REPEATED START
TX: 0xC0 ACK 0x5A ACK 0x00 ACK 0x00 ACK
I2C> [ 0xc0 0x95 0x00 0x00 0x00 0x00 0x00 0x00 0x00

I2C REPEATED START
TX: 0xC0 ACK 0x95 ACK 0x00 ACK 0x00 ACK 0x00 ACK 0x00 ACK 0x00 ACK 0x00 ACK
    0x00 ACK
I2C> [ 0xc0 0xa2 0x00 0x00 0x00 0x00

I2C REPEATED START
TX: 0xC0 ACK 0xA2 ACK 0x00 ACK 0x00 ACK 0x00 ACK 0x00 ACK
I2C> [ 0xc0 0xb7 0x92 ]

I2C REPEATED START
TX: 0xC0 ACK 0xB7 ACK 0x92 ACK
I2C STOP
I2C> p
Pull-up resistors: Disabled
I2C> w
Power supply: Disabled

I2C> i

This device complies with part 15 of the FCC Rules. Operation is subject to the following two conditions: (1) this device may not cause harmful interference, and (2) this device must accept any interference received, including interference that may cause undesired operation.


Bus Pirate 6
https://BusPirate.com/
Firmware main branch @ 8e51eb6 (2024-10-31T11:09:12Z)
RP2350B with 512KB RAM, 128Mbit FLASH
S/N: 2EAE869F8943242C
Storage:   0.10GB (FAT16 File System)

Configuration file: Loaded
Available modes: HiZ 1-WIRE UART HDUART I2C SPI 2WIRE DIO LED INFRARED
Active mode: I2C
 I2C speed: 100 kHz
 Clock stretching: OFF

Display format: Auto
Data format: 8 bits, MSB bitorder
Pull-up resistors: OFF
Power supply: OFF
Frequency generators: OFF

1.Vout  2.IO0   3.IO1   4.IO2   5.IO3   6.IO4   7.IO5   8.IO6   9.IO7   10.GND
OFF     SDA     SCL     -       -       -       -       -       -       GND
0.1V    0.1V    0.1V    0.0V    0.0V    0.0V    0.0V    0.0V    0.0V    GND

I2C>

Above is a console log of Bus Pirate v6 programming of the PLL for development testing.

… more to follow.