回路図
ATtiny85のpin1は、RESETとAnalog DIを兼ねている。pin1はレベルLoになるとリセット動作するので、抵抗分圧でレベルhiの時、DI動作させている。
ATtiny85パラメータ
チップ設定は、この様にした。Tiny4koled.hインストール
ライブラリを/srcに移したかったが、エラーが出たので諦めた。Tiny4koled.zipを同梱したので必要であればインストール。
スケッチ
///////////////////////////////////////////////////////////////////////
// si5351a oled(128x32) ATtiny85 VFO program ver.1.0
// Copyright(C)2023.JA2GQP.All rights reserved.
//
// 2023/8/27
// JA2GQP
//--------------------------------------------------------------------
// Function
// 1.STEP(10k,1k,100,10)
// 2.EEPROM memorry save/reload
// 3.Protection Operation At The Time Of Transmission
// 4.S-Meter
// 5.mono band
//////////////////////////////////////////////////////////////////////
#include "src/Rotary.h"
#include "src/si5351a21.h"
#include <Tiny4kOLED.h>
#include "font/Tlcdnums14x24.h"
#include "font/Tlabels.h"
#include "font/Tpixels.h"
#include <EEPROM.h>
////////////////////////////////
// Set Device
////////////////////////////////
#define SW1 (val < 545) && ( val >= 525) // SW1(1.70V)
#define SW2 (val < 710) && ( val >= 690) // SW2(2.23V)
#define SW3 (val < 790) && ( val >= 770) // SW3(2.50V)
#define SW4 (val < 840) && ( val >= 820) // SW4(2.65V))
#define SW_TX SW1
#define SW_STEP SW2
#define SW_MODE SW3
#define SW_RIT SW4
#define ENC_A PB1
#define ENC_B PB3
////////////////////////////////
// Set Device
////////////////////////////////
Rotary r = Rotary(ENC_B,ENC_A);
////////////////////////////////
// EEPROM Memory Address
////////////////////////////////
const byte Eep_Init = 0x00; // Eep Init(1byte*1)
const byte Eep_Band = 0x01; // Band(1byte*1)
const byte Eep_Mode = 0x02; // Mode(1byte*1)
const byte Eep_Freq = 0x10; // Frequency(4byte*8)
const byte Eep_Step = 0x30; // STEP(4byte*8)
////////////////////////////////
// frquency data
////////////////////////////////
const unsigned long FRQ_TBL[3] = {
// DEF LOW HI
7050000 ,7000000 ,7200000
};
//---- data offset -----
const byte DEF_F = 0;
const byte LOW_F = 1;
const byte HI_F = 2;
////////////////////////////////
// I/O Port
////////////////////////////////
const byte AD_IO = A0; // analog DI
const byte AD_SM = A2; // S-meter
//---- IF Frequency & Bfo data -----
const unsigned long IF_FREQ = 10700000L;
const unsigned long BFO_TBL[3] = {
// CW LSB USB
10700600,10701500,10698500
};
const byte CW = 0; // CW
const byte LSB = 1; // LSB
const byte USB = 2; // USB
////////////////////////////////
// etc
////////////////////////////////
const byte Int_End = 88; // Initial end code
const int LW_RIT = -5000; // RIT Lower Limit
const int HI_RIT = 5000; // RIT Upper Limit
////////////////////////////////
// Memory Assign
////////////////////////////////
unsigned long Vfo_Dat; // VFO Data
unsigned long Enc_Step; // STEP
int Rit_Dat; // RIT Data
unsigned int Val_Smeter = 0;
unsigned int Val_Smeterb = 1;
long Time_Passd; // int to hold the arduino miilis
byte Flg_eepWT = 0; // EEP Write Flag
byte Byt_Mode = LSB;
byte Flg_Over;
byte Flg_Tx = 0;
byte Flg_Rit = 0; // RIT Flag
long Lng_Wk; // Long Work
int Int_Wk; // Int Work
unsigned int val;
//---------- setup ----------------------------------------------------
void setup() {
pinMode(ENC_A, INPUT_PULLUP);
pinMode(ENC_B, INPUT_PULLUP);
GIMSK |= (1 << PCIE); // Enable pin change interrupt
PCMSK |= (1 << PCINT1) | (1 << PCINT3);
sei(); // INT Enable
oled.begin();
oled.clear();
oled.on();
//---------- EEPROM initialyze
if(EEPROM.read(Eep_Init) != Int_End){
delay(10);
eep_init();
}
val = analogRead(AD_IO);
delay(10);
if(SW_STEP){
delay(10);
eep_init();
fover_disp();
do{
val = analogRead(AD_IO);
delay(10);
}
while (SW_STEP);
}
eep_rdata();
step_disp(Enc_Step);
mode_disp(Flg_Tx,Byt_Mode);
}
//---------- main(Loop) -----------------------------------------------
void loop() {
////////////////////////
// reseive
////////////////////////
val = analogRead(AD_IO);
delay(10);
if(!SW_TX){
Flg_Tx = 0;
si5351a_enable(0x00); // si5351 output enable
//---------- RIT proc
if(Flg_Rit == 1){
si5351aSetFrequency(Vfo_Dat + IF_FREQ + Rit_Dat);
rit_disp(Rit_Dat);
}
//---------- normal proc(not RIT)
else{
band_check(); // range check
si5351aSetFrequency(Vfo_Dat + IF_FREQ);
freq_disp(Vfo_Dat); // frequency display
}
//---------- STEP SW check
val = analogRead(AD_IO);
delay(10);
if(SW_STEP){
enc_step();
step_disp(Enc_Step);
do{
val = analogRead(AD_IO);
delay(10);
}
while(SW_STEP);
}
//---------- MODE SW check
val = analogRead(AD_IO);
delay(10);
if(SW_MODE){
mode_set();
mode_disp(Flg_Tx,Byt_Mode);
do{
val = analogRead(AD_IO);
delay(10);
}
while(SW_MODE);
}
//---------- RIT SW check
val = analogRead(AD_IO);
delay(10);
if(SW_RIT){
if(Flg_Rit == 0)
Flg_Rit = 1;
else{
Flg_Rit = 0;
Rit_Dat = 0;
}
do{
val = analogRead(AD_IO);
delay(10);
}
while(SW_RIT);
}
}
////////////////////////
// trancemit
////////////////////////
else{
Flg_Tx = 1;
if(Flg_Over == 1){ // frequency limit over?
fover_disp();
si5351a_enable(0x05); // VFO and BFO off
}
else{
freq_disp(Vfo_Dat); // frequency display
si5351aSetFrequency(Vfo_Dat + IF_FREQ);
}
}
////////////////////////
// common
////////////////////////
mode_disp(Flg_Tx,Byt_Mode);
si5351aSetFrequency2(BFO_TBL[Byt_Mode]); // Bfo data out
//---------- s-meter
Val_Smeter = analogRead(AD_SM);
if ((abs(Val_Smeter - Val_Smeterb)) > 3){ // if needed draw S-meter
sm_disp();
Val_Smeterb = Val_Smeter;
}
//---------- Save to EEPROM
if(Flg_eepWT == 1){
if(Time_Passd+2000 < millis()){ // EEPROM auto write(2sec)
eep_wdata();
Flg_eepWT = 0;
}
}
}
//---------- Encorder procedure(INT) ----------------------
ISR(PCINT0_vect) {
unsigned char result = r.process();
if(Flg_Tx == 0){
if (result) {
if (result == DIR_CW){
Lng_Wk = Vfo_Dat + Enc_Step;
Int_Wk = Rit_Dat + Enc_Step;
}
else{
Lng_Wk = Vfo_Dat - Enc_Step;
Int_Wk = Rit_Dat - Enc_Step;
}
if(Flg_Rit == 1)
Rit_Dat = Int_Wk;
else{
Vfo_Dat = Lng_Wk;
Rit_Dat = 0;
}
Rit_Dat = constrain(Rit_Dat,LW_RIT,HI_RIT); // RIT range check
Flg_eepWT = 1;
Time_Passd = millis();
}
}
}
//---------- mode set -----------------------------------------------
void mode_set() {
if (Byt_Mode == 2)
Byt_Mode = 0;
else
Byt_Mode++;
}
//---------- band chek -----------------------------------------------
void band_check(){
if((Vfo_Dat >= FRQ_TBL[LOW_F])
&& (Vfo_Dat <= FRQ_TBL[HI_F]))
Flg_Over = 0;
else
Flg_Over = 1;
}
//---------- EEPROM Initialization ------------------------------------
void eep_init(){
int i;
for (i=0;i<64;i++) // 0 clear(64byte)
EEPROM.write(i, 0);
eep_write4(FRQ_TBL[DEF_F],Eep_Freq);
eep_write4(1000,Eep_Step);
EEPROM.write(Eep_Mode,LSB); // mode LSB
EEPROM.write(Eep_Init,Int_End); // Init end set(73)
}
//---------- EEPROM Dat Read -----------------------------------------
void eep_rdata(){
Vfo_Dat = eep_read4(Eep_Freq);
Enc_Step = eep_read4(Eep_Step);
Byt_Mode = EEPROM.read(Eep_Mode);
}
//---------- EEPROM Dat Write ----------------------------------------
void eep_wdata(){
eep_write4(Vfo_Dat,Eep_Freq);
eep_write4(Enc_Step,Eep_Step);
EEPROM.write(Eep_Mode,Byt_Mode); // mode LSB
}
//---------- Write EEPROM 4byte --------------------------------------
void eep_write4(long value,int address){
address += 3;
for(int i = 0;i < 4;i++){
byte toSave = value & 0xFF;
if(EEPROM.read(address) != toSave){
EEPROM.write(address,toSave);
}
value = value >> 8;
address--;
}
}
//---------- Read EEPROM 4byte ---------------------------------------
long eep_read4(int address){
long value = 0;
for(int i = 0;i < 4;i++){
value = value | EEPROM.read(address);
if( i < 3){
value = value << 8;
address++;
}
}
return value;
}
//---------- Encorede STEP -------------------------------------------
void enc_step() {
if (Enc_Step == 10000)
Enc_Step = 10; // 1000 Hz, round to XX.XXX.000
else
Enc_Step = Enc_Step * 10;
}
//---------- S-Meter Display -----------------------------------------
void sm_disp() {
uint8_t a = 0;
uint8_t m = 0;
a = (Val_Smeter + 3) / 113; // 1024 / 9 characters
oled.setFont(&Tpixels);
oled.setCursor(25, 3);
for (m = 0; m < a; m++)
if (m < 6)
oled.print('7'); // '5' - hollow rectangle, 6px
else
oled.print('8'); // '6' - filled rectangle, 6px
for (m = a; m < 9; m++)
oled.print('.'); // '.' 1px
}
//---------- STEP Display --------------------------------------------
void step_disp(unsigned long stp) {
oled.setCursor(109, 3);
oled.setFont(&Tlabels);
if (stp == 10)
oled.print("5"); // 10
else if(stp == 100)
oled.print("7"); // 100
else if(stp == 1000)
oled.print("8"); // 1k
else
oled.print("9"); // 10k
}
//---------- Mode display ----------------------------------------------
void mode_disp(byte flg,byte mod) {
oled.setCursor(2, 3);
oled.setFont(&Tlabels);
if(flg == 1)
oled.print("1"); // "1" is "TX" in labels.h
else{
if(mod == 0)
oled.print("2"); // "2" is "CW" in labels.h
else if(mod == 1)
oled.print("3"); // "3" is "LSB"
else if(mod == 2)
oled.print("4"); // "4" is "USB"
}
}
//---------- Frequency renge over --------------------------------------
void fover_disp() {
oled.setFont(&Tlcdnums14x24);
oled.setCursor(1, 0);
oled.print("--.---.--");
}
//---------- RIT Display ---------------------------------------------
void rit_disp(int rit_data) {
unsigned int ri,rit;
oled.setFont(&Tlcdnums14x24);
oled.setCursor(1, 0);
oled.print("::::");
if (rit_data < 0)
oled.print('-');
else
oled.print('+');
rit = abs(rit_data);
ri = rit / 1000;
oled.print(ri);
oled.print('.');
ri = (rit % 1000) / 10;
if (ri < 10)
oled.print('0');
oled.print(ri);
}
//---------- Frequency Display ---------------------------------------
void freq_disp(uint32_t sf_rx) {
uint16_t fr;
oled.setFont(&Tlcdnums14x24);
oled.setCursor(1, 0);
fr = sf_rx / 1000000;
if (fr < 10)
oled.print(':'); // ':' is changed to ' '
oled.print(fr);
oled.print('.');
fr = (sf_rx % 1000000) / 1000;
if (fr < 100)
oled.print('0');
if (fr < 10)
oled.print('0');
oled.print(fr);
oled.print('.');
fr = (sf_rx % 1000) / 10;
if (fr < 10)
oled.print('0');
oled.print(fr);
}
mono BAND用で、96%のメモリー消費である。
4 件のコメント:
素晴らしい作品の公開ありがとうございます。
現在、本作品に取り組んでおります。
ちょっとわからないことがあり、二点ほどご確認させください。
1.
まず、本作品を焼くときのFUSE設定ですが
8MHz(Internal)の時に LFUSEですがデフォルトのL:62ではなくL:E2の設定でCKDIV8を無効にされた設定で焼かれたのでしょうか?
(3.3V動作ですが16MHz(PLL)でL:F1も試しているところです)
L:62だとちょっともっさりした動きでしたので確認させてください。
2.
周波数の表示についてですが、以前のsi5351a VFO Ver1.1では
freq+ifshift として周波数表示がキャリア周波数のところに
になっていたと認識していましたが最近のコードでは
IF_FREQとなっていて、表示周波数に違和感がありました。
本コードも、BFO_TBL[Byt_Mode]のオフセットにするようにすれば
最近の無線機と同じようになるのでは?と思ったのですが
これまでのコードから変更された理由は何かあるのでしょうか?
このoled85VFOのコードは回路図もコードも洗練されているので
モノバンドではいろいろシンプルに楽しめそうに思っております。
よろしくお願いします。
7n4mis OM
1.アセンブラーではないので、特にfuseといった概念は有りません。
ただ、chip設定画面でパラメータを設定するのみです。この結果がfuseに反映されるはずです。パラメータ設定した内容は画面コピーの通りです。
2.周波数セ帝に関して考え方の相違です。基本的な考え方は、キャリア周波数を表示すべきだと考えてます。例えばLSB/USB/CWとモードを変えた時、受信周波数が変わること自体不自然です。最近の無線機はいろんな設定ができ、キャリア周波数での表示も可能です。
Dds arduino nano
This VFO is neither DDS nor nano.
If you want to customize it to your specifications, please do it yourself.
As I wrote at the top of the blog, I don't customize it.
コメントを投稿