
Motivation
![]() |
TEA 5767 FM radio module |
In my first experiments with TEA5767 in a breadboard I used the libraries from Simon Monk and Andy Karpov. Their code helped me a lot to understand how to control the TEA5767.
Then, to make a more complete application, I felt the need to organize my code a little bit and, with the help of the TEA5767 Application Note, I encapsulated the TEA5767 commands in several convenient methods.
Data Structure
To send or receive TEA5767 commands there are always five bytes involved. Even if you need to send a single bit, all five bytes must be resent. So I kept transmission and reception data bytes as instance attributes of the object, as seen below.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
class TEA5767N { | |
private: | |
//... | |
byte transmission_data[5]; | |
byte reception_data[5]; | |
//... | |
public: | |
//... | |
}; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
TEA5767N::TEA5767N() { | |
Wire.begin(); | |
initializeTransmissionData(); | |
} | |
void TEA5767N::initializeTransmissionData() { | |
transmission_data[FIRST_DATA] = 0; | |
transmission_data[SECOND_DATA] = 0; | |
transmission_data[THIRD_DATA] = 0xB0; | |
transmission_data[FOURTH_DATA] = 0x10; | |
transmission_data[FIFTH_DATA] = 0x00; | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
void TEA5767N::mute() { | |
setSoundOff(); | |
transmitData(); | |
} | |
void TEA5767N::setSoundOff() { | |
transmission_data[FIRST_DATA] |= 0b10000000; | |
} | |
void TEA5767N::transmitData() { | |
Wire.beginTransmission(TEA5767_I2C_ADDRESS); | |
for (int i=0 ; i<6 ; i++) { | |
Wire.write(transmission_data[i]); | |
} | |
Wire.endTransmission(); | |
delay(100); | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
void TEA5767N::setSoundOn() { | |
transmission_data[FIRST_DATA] &= 0b01111111; | |
} |
And as said earlier, there are also five bytes that can be read from the TEA5767 module. They are stored in the reception_data array and depending on the operation there is the need to update the transmission_data array too so they can be in sync. Here is an example:
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
void TEA5767N::loadFrequency() { | |
readStatus(); | |
transmission_data[FIRST_DATA] = (transmission_data[FIRST_DATA] & 0xC0) | (reception_data[FIRST_DATA] & 0x3F); | |
transmission_data[SECOND_DATA] = reception_data[SECOND_DATA]; | |
} | |
void TEA5767N::readStatus() { | |
Wire.requestFrom (TEA5767_I2C_ADDRESS, 5); | |
if (Wire.available ()) { | |
for (int i = 0; i < 5; i++) { | |
reception_data[i] = Wire.read(); | |
} | |
} | |
delay(100); | |
} |
Algorithms
Reading the TEA5767 application note, you´ll find formulas and algorithms to set and read the station frequency. For example, to select a station frequency it´s necessary first calculate the optimal hi / lo injection which consists in select the (frequency - delta) and (frequency + delta), and choose the one that offers a better signal level. Delta is a constant that represents 450 kHz.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
void TEA5767N::selectFrequency(float frequency) { | |
calculateOptimalHiLoInjection(frequency); | |
transmitFrequency(frequency); | |
} | |
void TEA5767N::calculateOptimalHiLoInjection(float freq) { | |
byte signalHigh; | |
byte signalLow; | |
setHighSideLOInjection(); | |
transmitFrequency((float) (freq + 0.45)); | |
signalHigh = getSignalLevel(); | |
setLowSideLOInjection(); | |
transmitFrequency((float) (freq - 0.45)); | |
signalLow = getSignalLevel(); | |
hiInjection = (signalHigh < signalLow) ? 1 : 0; | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
void TEA5767N::transmitFrequency(float frequency) { | |
setFrequency(frequency); | |
transmitData(); | |
} | |
void TEA5767N::setFrequency(float _frequency) { | |
frequency = _frequency; | |
unsigned int frequencyW; | |
if (hiInjection) { | |
setHighSideLOInjection(); | |
frequencyW = 4 * ((frequency * 1000000) + 225000) / 32768; | |
} else { | |
setLowSideLOInjection(); | |
frequencyW = 4 * ((frequency * 1000000) - 225000) / 32768; | |
} | |
transmission_data[FIRST_DATA] = ((transmission_data[FIRST_DATA] & 0xC0) | ((frequencyW >> 8) & 0x3F)); | |
transmission_data[SECOND_DATA] = frequencyW & 0XFF; | |
} |
Searching stations
Before actually starting a search, there are a few things that you have to set up: if the search is up or down, the signal level a station must have to stop the search, and is recommended to mute the radio so that users don't hear that station change annoying sound.
To choose up or down search direction is also just change one bit in the transmission_data array.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
void TEA5767N::setSearchUp() { | |
transmission_data[THIRD_DATA] |= 0b10000000; | |
} | |
void TEA5767N::setSearchDown() { | |
transmission_data[THIRD_DATA] &= 0b01111111; | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
#define LOW_STOP_LEVEL 1 | |
#define MID_STOP_LEVEL 2 | |
#define HIGH_STOP_LEVEL 3 |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
void TEA5767N::setSearchLowStopLevel() { | |
transmission_data[THIRD_DATA] &= 0b10011111; | |
transmission_data[THIRD_DATA] |= (LOW_STOP_LEVEL << 5); | |
} | |
void TEA5767N::setSearchMidStopLevel() { | |
transmission_data[THIRD_DATA] &= 0b10011111; | |
transmission_data[THIRD_DATA] |= (MID_STOP_LEVEL << 5); | |
} | |
void TEA5767N::setSearchHighStopLevel() { | |
transmission_data[THIRD_DATA] &= 0b10011111; | |
transmission_data[THIRD_DATA] |= (HIGH_STOP_LEVEL << 5); | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
byte TEA5767N::searchNext() { | |
byte bandLimitReached; | |
if (isSearchUp()) { | |
selectFrequency(readFrequencyInMHz() + 0.1); | |
} else { | |
selectFrequency(readFrequencyInMHz() - 0.1); | |
} | |
//Turns the search on | |
transmission_data[FIRST_DATA] |= 0b01000000; | |
transmitData(); | |
while(!isReady()) { } | |
//Read Band Limit flag | |
bandLimitReached = isBandLimitReached(); | |
//Loads the new selected frequency | |
loadFrequency(); | |
//Turns de search off | |
transmission_data[FIRST_DATA] &= 0b10111111; | |
transmitData(); | |
return bandLimitReached; | |
} |
. Add or subtract 100 kHz from current station (so search doesn´t find current station again), depending on search direction;
. Sets the search bit;
. Waits for radio search;
. When search is ready, checks whether band limit has been reached or not;
. Update transmission_data array with the new selected frequency;
. Turn off the search bit;
. Returns band limit status.
Search_next() does not mute before search and is useful for debug purpose. To mute before search you can call mute() method yourself or you can use the convenient searchNextMuting() method.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
byte TEA5767N::searchNextMuting() { | |
byte bandLimitReached; | |
mute(); | |
bandLimitReached = searchNext(); | |
turnTheSoundBackOn(); | |
return bandLimitReached; | |
} |
Two other useful search methods are startsSearchFromBeginning() and startsSearchFromEnd(). What they do is set the direction and start frequency before call searchNext().
I used these methods in a project to scan and record all avalilable stations. That project, as I said, will be my next post here.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
byte TEA5767N::startsSearchFromBeginning() { | |
setSearchUp(); | |
return startsSearchFrom(87.0); | |
} | |
byte TEA5767N::startsSearchFromEnd() { | |
setSearchDown(); | |
return startsSearchFrom(108.0); | |
} | |
byte TEA5767N::startsSearchFrom(float frequency) { | |
selectFrequency(frequency); | |
return searchNext(); | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
byte TEA5767N::startsSearchMutingFromBeginning() { | |
byte bandLimitReached; | |
mute(); | |
bandLimitReached = startsSearchFromBeginning(); | |
turnTheSoundBackOn(); | |
return bandLimitReached; | |
} | |
byte TEA5767N::startsSearchMutingFromEnd() { | |
byte bandLimitReached; | |
mute(); | |
bandLimitReached = startsSearchFromEnd(); | |
turnTheSoundBackOn(); | |
return bandLimitReached; | |
} |
Thanks to the ability to write this library in C++, an object oriented language, I could write more cohesive methods that are easy to use by other client software. This way, using this library you can build your system step-by-step as you try the module features. Separating methods in public and private members also helps to understand how the library is supposed to be used.
Another good practice you can find in this library is the careful choice of the variables and methods names. They reveal the intention, what´s the variable role and what the methods do.
I show how I used this library with Arduino, TEA5767 FM module and the LCD Keypad Shield to build a complete radio application.
Thank you!
Thank you for reading!
Please let me know if this library somehow helped you in your projects.
To follow my updates to this library and my other projects: