Leia este documento em Português
This is a library for the SI4844, BROADCAST ANALOG TUNING DIGITAL DISPLAY AM/FM/SW RADIO RECEIVER, IC from Silicon Labs for the Arduino development environment. This library is intended to provide an easier interface for controlling the SI4844.
By Ricardo Lima Caratti, Oct, 2019.
Attention: Documentation Under construction.
- Your support is important
- About the SI4844 Architecture
- Terminology
- Library Installation
- Hardware Requirements and Setup
- API Documentation
- Defined Data Types and Structures
- Public Methods
- setup
- reset
- setBand
- changeVolume
- setVolume
- getStatus
- getFirmware
- getFrequency
- hasStatusChanged
- resetStatus
- getBandMode
- getStereoIndicator
- getStatusBCFG0
- getStatusBCFG1
- getStatusStereo
- getStatusStationIndicator
- getStatusInformationReady
- getStatusHostPowerUp
- getStatusHostReset
- getStatusBandMode
- getStatusBandIndex
- getStatusCTS
- getFirmwareErr
- getFirmwareCTS
- getFirmwarePartNumber
- getFirmwareMajorRevision
- getFirmwareMinorRevision
- getFirmwareComponentMajorRevision
- getFirmwareComponentMinorRevision
- getFirmwareChipRevision
- References
- Examplel
- Videos
If you would like to support this library development, consider joining this project via Github. Alternatively, make suggestions on features you would like available in this library. Thank you!
The Si4844 is an analog-tuned digital-display AM/FM/SW radio receiver. It has an analog-tune while frequency, band, and setero/mono information can be displayed on LCD. It works with a I2C protocol that allows a microcontroller send command and receive data.
See more about SI4844 on BROADCAST ANALOG TUNING DIGITAL DISPLAY AM/FM/SW RADIO RECEIVER
Term | Description |
---|---|
Arduino Libraries | Libraries are files written in C or C (.c, .cpp) which provide your sketches with extra functionality. The SI4844 Library provides extra functionalities to make easier the Arduino deal with Si4844 device |
IDE | Integrated Development Environment |
Sketch | Name that Arduino environment uses for a program |
ATDD | Analog Tune Digital Display. Sometimes used to refer the Si4844 device |
interrupt | In this context, it is a Arduino Resource. Allows important tasks to be performed regardless of the flow of your program |
C | A object-oriented programming (OOP) language. It is a superset of the C language with an additional concept of "classes." |
programming guide | In this context it refers to Si48XX ATDD PROGRAMMING GUIDE |
POC | Proof of Concept |
You can install the library is via the Arduino Library Manager. Go to the menu Sketch > Include Library > Manage Libraries..., and in the search box, type "PU2CLR SI4844".
The images below show that actions
This library has been written for the Arduino platform and has been successfully tested on Pro Mini. I beleave it will work on any other Arduino with I2C support.
The SI4844 is a 3.3 V only part. If you are not using a 3.3 V version of Arduino, you have to use a kind of converter.
To use this labrary you need to build a radio based on SI4844 coneccted to Arduino. The schematic and photos below show the hardware and setup requirments for start using this library.
The signal amplifier was not required for this test.
Original Source: Raymond Genovese, May 26, 2016 - How to Build an Arduino-Controlled AM/FM/SW Radio
The document BROADCAST ANALOG TUNING DIGITAL DISPLAY AM/FM/SW RADIO RECEIVER, chapter two, page 11, has a typical application schematic.
Attention: If you are not using an Arduino Pro Mini, pay attention on the appropriated Arduino pinout to select the correct interrupt (IRQ), RST, SDIO and SCLK pins.
Board | Interrupr (IRQ) Pins | I2C / TWI pins |
---|---|---|
328-based (Nano, Mini or Uno) |
D2 and D3 | A4 (SDA/SDIO), A5 (SCL/SCLK) |
Mega | 2, 3, 18, 19, 20 and 21 | 20 (SDA/SDIO), 21 (SCL/SCLK) |
32u4-based (Micro, Leonardo or Yum) |
0, 1, 2, 3 and 7 | 2 (SDA/SDIO), 3 (SCL/SCLK) |
Zero | all digital pins except pin 4 | D8 (SDA/SDIO) and D9 (SCL/SCLK) |
Due | all digital pins | 20 (SDA/SDIO), 21 (SCL/SCLK) |
101 | all digital pins. Only pins 2, 5, 7, 8, 10, 11, 12, 13 work with CHANGE |
The figure below shows that schematic
Source: Silicon Labs (Si4840/44-A10)
Parts list used by the first schematic
The table below is based on Raymond Genovese, May 26, 2016 - How to Build an Arduino-Controlled AM/FM/SW Radio .
Part | Description |
---|---|
(*1) B1 | ferrite bead 2.5 kOhm (100 mHz) B1, ferrite bead 2.5 kOhm (100 mHz), 81-BLM18BD252SZ1D |
C1,C2,C5 | 4.7 uf non polarized capacitor |
C3,C4 | 22 pf non polarized capacitor |
C6,C7,C9 | .1 uf non polarized capacitor |
(*2) C8 | 4.7 uf non polarized capacitor (the orinal value is 47uF. I used 4.7uF) |
C10, (*1) C11 | .47 uf non polarized capacitor |
(*1) C12, (*1) C14 | 33 nf non polarized capacitor |
C13 | 33 pf non polarized capacitor |
(*1) C15 | 10 pf non polarized capacitor |
IC1 | Si4844-A10 radio receiver |
(*1) Q1 | SS9018 NPN transistor |
R1, R2 | 2.2K |
(*1) R3 | 1K |
R4, (*1) R7 | 100K |
(*1) R5 | 10 Ohms |
(*1) R6 | 120K |
R8 | 100 Ohms |
L1 | 270 nH Inductor (0,270 uH) |
VR1 | 100K linear potentiometer |
Y1 | 32.768 kHz crystal |
ANT1 | ferrite antenna |
ANT2 | telescopic/whip antenna |
Notes | *1. not used for this project; *2. the value used is 4.7uF and not 47uF as suggested by the original circuit. |
It was a bit hard to solder the Si4844 on adapter. However, by using a electronic magnifier it was possible.
This labrary was developed in C . To use it, you must declare in your Sketch a variable of the class SI4844. The code below show that.
#include <SI4844.h>
#include <Wire.h>
// Arduino Pin (tested on pro mini)
#define INTERRUPT_PIN 2
#define RESET_PIN 12
#define DEFAULT_BAND 4
SI4844 si4844;
void setup() {
// Initiate and connect the device ATDD (SI4844) to Arduino
si4844.setup(RESET_PIN, INTERRUPT_PIN, DEFAULT_BAND);
}
void loop() {
// if something changed on ATDD (SI4844), do something
if (si4844.hasStatusChanged())
{
Serial.print("Band Index: ");
Serial.print(si4844.getStatusBandIndex());
Serial.print(" - ");
Serial.print(si4844.getBandMode());
Serial.print(" - Frequency: ");
Serial.print(si4844.getFrequency(),0);
Serial.print(" KHz");
Serial.print(" - Stereo ");
Serial.println(si4844.getStereoIndicator());
}
}
See Proof of Concept for SI4844 Arduino Library to get the entire exemple.
To make the SI4844 device easier to deal, some defined data types were built to handle byte and bits responses.
/*
* The structure below represents the four bytes response got by command ATDD_GET_STATUS
* See Si48XX ATDD PROGRAMMING GUIDE, pages 14 and 15
*/
typedef struct
{
byte BCFG0 : 1; // Bit 0
byte BCFG1 : 1; // bit 1
byte STEREO : 1; // bit 2
byte STATION : 1; // bit 3
byte INFORDY : 1; // bit 4
byte HOSTPWRUP : 1; // bit 5
byte HOSTRST : 1; // bit 6
byte CTS : 1; // bit 7
byte BANDIDX : 6; // Form bit 0 to 5
byte BANDMODE : 2; // From bit 6 to 7
byte d2 : 4; // Frequency digit 2
byte d1 : 4; // Frequency digit 1
byte d4 : 4; // Frequency digit 4
byte d3 : 4; // frequency digit 3
} si4844_get_status;
/*
* Uses a C language feature to represent two way for the 4 response bytes (status) sent by the ATDD_GET_STATUS.
* It is needed to undertand the C language union concept.
* See Si48XX ATDD PROGRAMMING GUIDE, pages 14 and 15
*/
typedef union {
si4844_get_status refined;
byte raw[4];
} si4844_status_response;
// English:
// GET_REV structure. The structure below represents 9 bytes response for GET_REV command.
// STATUS and RESP1 to RESP8. See Si48XX ATDD PROGRAMMING GUIDE; AN610, page 22.
// Portuguese:
typedef struct
{
byte RESERVED : 6; // Bit 0 to 5
byte ERR : 1; // bit 6
byte CTS : 1; // bit 2
byte PN; // Final 2 digits of Part Number (HEX).
byte FWMAJOR; // Firmware Major Revision (ASCII).
byte FWMINOR; // Firmware Minor Revision (ASCII).
byte CMPMAJOR; // Component Major Revision (ASCII).
byte CMPMINOR; // Component Minor Revision (ASCII).
byte CHIPREV; // Chip Revision (ASCII).
} si4844_firmware_info;
typedef union {
si4844_firmware_info refined;
byte raw[9];
} si4844_firmware_response;
/*
* Initiate the SI4844 instance and connect the device (SI4844) to Arduino.
* Calling this library should be the first thing to do to control the SI4844.
*
* @param resetPin arduino pin used to reset the device
* @param interruprPin arduino pin used to handle interrupr
* @param defaultBand band that the radio should start
*/
void setup(unsigned int, unsigned int, byte)
Example:
si4844.setup(12, 2, 4);
/*
* reset
* English
* See pages 7, 8, 9 and 10 of the programming guide.
*/
void reset(void )
Example:
si4844.reset();
/*
* Set the radio to a new band.
* See Table 8. Pre-defined Band Table in Si48XX ATDD PROGRAMMING GUIDE; AN610; pages 17 and 18
*/
void setBand(byte);
Example:
si4844.setBand(4); // FM
/*
* Up or down the sound volume level/
* @param char ' ' up and '-' down
*/
void changeVolume(char);
Example:
si4844.changeVolume(' ');
/*
* Set the sound volume level.
* @param byte volumeLevel (domain: 0 to 63)
*/
setVolume(byte level)
Example:
si4844.setVolume(55);
/*
* Get tune freq, band, and others information, status of the device.
* Use this method only if you want to deal with that information by yourself.
* This library has other methods to get that information easier.
*
* @return a pointer to a structure type si4844_status_response
*/
si4844_status_response *getStatus(void);
Example:
si4844.getStatus();
/*
* Get part number, chip revision, firmware, patch, and component revision numbers.
* You do not need to call this method. It is executed just once at setup methos.
* There are other methods that give you that information.
* See page 22 of programming guide.
*
* @return a pointer to a structure type with the part number, chip revision,
* firmware revision, patch revision, and component revision numbers.
*/
si4844_firmware_response *getFirmware(void);
Example:
si4844.getFirmware();
/*
* Get the current frequency of the radio in KHz.
* For example: FM, 103900 KHz (103.9 MHz);
* SW, 7335 KHz (7.34 MHz, 41m)
*
* @return float current frequency in KHz.
*/
float getFrequency(void);
Example:
Serial.print("Frequency: ");
Serial.print(si4844.getFrequency(),0);
/*
* Check if the SI4844 has its status changed. If you move the tuner, for example,
* the status of the device is changed.
*
* return true or false
*/
bool hasStatusChanged(void);
Example:
if (si4844.hasStatusChanged())
{
Serial.print(" - Frequency: ");
Serial.print(si4844.getFrequency(),0);
Serial.print(" KHz");
Serial.print(" - Stereo ");
Serial.println(si4844.getStereoIndicator());
}
void resetStatus(void);
/*
* Get te current band(FM/AM/SW)
* @return char * "FM", "AM" or "SW".
*/
inline String getBandMode()
/*
* Get a string (char *) On or Off indicating stereo situation.
* return char * "On" or "Off"
*/
inline String getStereoIndicator()
/*
* See Si48XX ATDD PROGRAMMING GUIDE, AN610, pages 15 and 16
*/
inline unsigned getStatusBCFG0()
/*
* See Si48XX ATDD PROGRAMMING GUIDE, AN610, pages 15 and 16
*/
inline unsigned getStatusBCFG1()
/*
* Return a integer indicating the stereo status (0 = Off and 1 = On).
* See Si48XX ATDD PROGRAMMING GUIDE, AN610, pages 15 and 16
*/
inline unsigned getStatusStereo()
/*
* See Si48XX ATDD PROGRAMMING GUIDE, AN610, pages 15 and 16
*/
inline unsigned getStatusStationIndicator()
/*
* See Si48XX ATDD PROGRAMMING GUIDE, AN610, pages 15 and 16
*/
inline unsigned getStatusInformationReady()
/*
* See Si48XX ATDD PROGRAMMING GUIDE, AN610, pages 15 and 16
*/
inline unsigned getStatusHostPowerUp()
/*
* See Si48XX ATDD PROGRAMMING GUID, AN610, pages 15 and 16
*/
inline unsigned getStatusHostReset()
/*
* Return an integer (0 - FM; 1 = AM and 2 = SW)
* See Si48XX ATDD PROGRAMMING GUIDE, AN610, pages 15 and 16
*/
inline unsigned getStatusBandMode()
/*
* return an integer with the currente band index in use.
* Set Table 13. Pre-defined Band Table in Si48XX ATDD PROGRAMMING GUID, AN610, pages 39 and 40
*/
inline unsigned getStatusBandIndex()
/*
* See Si48XX ATDD PROGRAMMING GUIDE, AN610, pages 15 and 16
*/
inline unsigned getStatusCTS()
inline unsigned getFirmwareErr()
inline unsigned getFirmwareCTS()
/*
* Get Firmware Final 2 digits of Part Number (HEX).
*/
inline unsigned getFirmwarePartNumber()
See example below
/*
* Get Firmware Major Revision (ASCII).
*/
inline unsigned getFirmwareMajorRevision()
See example below
/*
* Get Firmware Minor Revision (ASCII).
*/
inline unsigned getFirmwareMinorRevision()
See example below
/*
* Get Firmware Component Major Revision (ASCII).
*/
inline unsigned getFirmwareComponentMajorRevision()
See example below
/*
* Get Firmware Component Minor Revision (ASCII).
*/
inline unsigned getFirmwareComponentMinorRevision()
See example below
/*
* Chip Revision (ASCII).
*/
inline unsigned getFirmwareChipRevision()
Example:
Serial.println("\nSI4844 - Firmware information\n");
si4844.getFirmware();
Serial.print("Final 2 digits of Part Number..: ");
Serial.println(si4844.getFirmwarePartNumber());
Serial.print("Firmware Major Revision........: ");
Serial.println(si4844.getFirmwareMajorRevision());
Serial.print("Firmware Minor Revision........: ");
Serial.println(si4844.getFirmwareMinorRevision());
Serial.print("Component Major Revision.......: ");
Serial.println(si4844.getFirmwareComponentMajorRevision());
Serial.print("Component Minor Revision.......: ");
Serial.println(si4844.getFirmwareComponentMinorRevision());
Serial.print("Chip Revision..................: ");
Serial.println(si4844.getFirmwareChipRevision());
Serial.println("*****************************");