2024年12月12日木曜日

6m AM VFO

6m AM TRX専用のVFOを作った。ATmega1614、si5351とoled表示器を使ったVFO。6m AM TRXはダブルス―パー方式を採用したので、si5351の全てのチャンネルを使うことにした。また、新しい機能として受信SHIFTを追加した。SHIFT機能は、受信周波数を-1.5k(LSB)、+1.5k(USB)、SHIFTゼロ(ブランク)をSHIFT SWで切り替えて行う仕様である。







回路図

















 

スケッチ

//////////////////////////////////////////////////////////////////////
//  si5351a oled(128x32) VFO program ver.1.0
//    Copyright(C)2019.JA2GQP.All rights reserved.
//
//                                                2024/12/11
//                                                  JA2GQP
//--------------------------------------------------------------------
//  Function
//    1.STEP(10k,1k,100)
//    2.Ofset(0,-1500,+1500)
//    3.Automatic memory
//    44.Protection Operation At The Time Of Transmission
//    5.S-Meter
//////////////////////////////////////////////////////////////////////

#include "src/Rotary.h"             // http://www.buxtronix.net/2011/10/rotary-encoders-done-properly.html
#include "src/si5351a.h"            // https://github.com/etherkit/Si5351Arduino, v2.1.0
#include <Tiny4kOLED.h>
#include "font/Tlcdnums14x24.h"
#include "font/Tlabels.h"
#include "font/Tpixels.h"
#include <EEPROM.h>

//---------- I/O Port -------------------------------

#define SW_STEP 0                   // STEP SW
#define SW_TX   1                   // TX/RX
#define ENC_A   2                   // encorder A
#define ENC_B   3                   //          B
#define SW_RIT  4                   // RIT SW
#define SW_OFT  5                   // OFFSET SW
#define AD_SM   10                  // S-meter AD

//---------- Set Device -----------------------------

Rotary r = Rotary(ENC_B, ENC_A);              

//---------- EEPROM Memory Address ------------------

#define Eep_Init    0x00            // Eep Init(1byte)
#define Eep_Oflg    0x02            // Offset flag(1byte)
#define Eep_Odat    0x04            // Offset data(4byte)
#define Eep_Freq    0x10            // Frequency(4byte)
#define Eep_Step    0x30            // STEP(4byte)

//---------- frquency data --------------------------

//---- frequency limit offset
#define DEF_F   50600000            // default frequency
#define LOW_F   50000000
#define HI_F    51000000

//---- receiver IF offset
#define IF_FREQ 10700000
#define RX_FIL  455000
const unsigned long RX_LO =   IF_FREQ - RX_FIL;          

//---- encorder STEP
#define STEP_100  100               // STEP 100Hz
#define STEP_1k   1000              //      1k
#define STEP_10k  10000             //      10k

//---- RIT limit
#define LW_RIT  -5000               // RIT low -5k
#define HI_RIT   5000               // RIT high 5k

//---- OFFSET limit
#define LW_OFT  -1500               // OFFSET low -1500
#define HI_OFT   1500               // OFFSET high 1500

//---------- etc ------------------------------------

const byte Int_End = 73;             // Initial end code

//---------- Memory Assign --------------------------

unsigned long Vfo_Dat;                // VFO Data
int Rit_Dat = 0;                      // RIT Data
int Oft_Dat = 0;                      // OFFSET Data
unsigned long Enc_Step;               // STEP
unsigned int Val_Smeter = 0;
unsigned int Val_Smeterb = 1;

unsigned long Time_Passd;             // int to hold the arduino miilis since startup
unsigned long Time_smeter;            // smeter scan Time
byte Flg_eepWT = 0;                   // EEP Write Flag

byte Flg_Over = 0;
byte Flg_Tx = 0;
byte Flg_Rit = 0;                     // RIT Flag
byte Flg_Oflg = 0;                    // OFFSET Flag
long Lng_Wk;                          // Long Work
int  Int_Wk;                          // Int Work

//----------  setup  ----------------------------------------------------

void setup() {
  pinMode(SW_STEP, INPUT_PULLUP);
  pinMode(SW_TX, INPUT_PULLUP);          
  pinMode(SW_RIT, INPUT_PULLUP);          
  pinMode(SW_OFT, INPUT_PULLUP);          

  attachInterrupt(ENC_A, rotary_encoder, CHANGE);
  attachInterrupt(ENC_B, rotary_encoder, CHANGE);

  oled.begin();
  oled.clear();
  oled.on();

  if(EEPROM.read(Eep_Init) != Int_End){   // Eep initialaz
    delay(10);
    eep_init();
  }
  else{
    if(digitalRead(SW_STEP) == LOW){
      delay(10);
      eep_init();
      fover_disp();
      while (digitalRead(SW_STEP) == LOW);
    }
  }
  eep_rdata();
  freq_disp(Vfo_Dat + Oft_Dat);       // frequency display
  oft_disp(Flg_Oflg);
  step_disp();
}

//----------  main  ----------------------------------------------------

void loop() {
//---------- receive ----------------------
  if(digitalRead(SW_TX) == HIGH){          
    Flg_Tx = 0;
    si5351a_enable(0x02);                  // clk0,clk2=enable clk1=disable
    si5351aSetFrequency2(RX_LO);           // RX LO set

//----------  RIT proc
    if (Flg_Rit == 0){
      si5351aSetFrequency0(Vfo_Dat - IF_FREQ + Oft_Dat);
      freq_disp(Vfo_Dat + Oft_Dat);                  // frequency display
//      oft_disp(Flg_Oflg);
    }      
    else{
      si5351aSetFrequency0(Vfo_Dat - IF_FREQ + Rit_Dat + Oft_Dat);
      rit_disp(Rit_Dat);
    }  

//---------- STEP SW check
    if (digitalRead(SW_STEP) == LOW) {    
      enc_step();
      step_disp();
      while (digitalRead(SW_STEP) == LOW);
    }

//---------- RIT SW check
    if (digitalRead(SW_RIT) == LOW) {
      if(Flg_Rit == 0)
        Flg_Rit = 1;
      else{
        Flg_Rit = 0;
        Rit_Dat = 0;
      }
      while (digitalRead(SW_RIT) == LOW);
    }

//---------- OFFSET SW check
    if (digitalRead(SW_OFT) == LOW) {
      switch(Flg_Oflg){
        case 0:
          Oft_Dat = 0;
          freq_disp(Vfo_Dat + Oft_Dat);        // frequency display
          oft_disp(Flg_Oflg);
          EEPROM.write(Eep_Oflg,Flg_Oflg);     // offset flg EEPROM save  
          Flg_Oflg++;
          eep_write4(Oft_Dat,Eep_Odat);
          break;
        case 1:
          Oft_Dat = LW_OFT;
          freq_disp(Vfo_Dat + Oft_Dat);        // frequency display
          oft_disp(Flg_Oflg);
          EEPROM.write(Eep_Oflg,Flg_Oflg);     // offset flg EEPROM save  
          Flg_Oflg++;
          eep_write4(Oft_Dat,Eep_Odat);
          break;
        case 2:
          Oft_Dat = HI_OFT;
          freq_disp(Vfo_Dat + Oft_Dat);        // frequency display
          oft_disp(Flg_Oflg);
          EEPROM.write(Eep_Oflg,Flg_Oflg);     //  offset flg EEPROM save  
          Flg_Oflg = 0;
          eep_write4(Oft_Dat,Eep_Odat);
          break;
      }
      while (digitalRead(SW_OFT) == LOW);
    }

  }

//---------- send  -------------------------
  else{                                    
    Flg_Tx = 1;
    band_check(Vfo_Dat);                    // range check
   
    if(Flg_Over == 0){                      // Within transmittable range?
      freq_disp(Vfo_Dat);                   // frequency display
      si5351a_enable(0x05);                  // si5351 output enable
      si5351aSetFrequency1(Vfo_Dat);
    }
    else{                                   // Out of range
      si5351a_enable(0x02);                  // si5351 output enable
      fover_disp();                       // Display of over range
    }
  }

//---------- common ------------------------

//---------- s meter
  if(Time_smeter+100 < millis()){
    Val_Smeter = analogRead(AD_SM);
    if ((abs(Val_Smeter - Val_Smeterb)) > 3){ // if needed draw S-meter
      sm_disp();
      Val_Smeterb = Val_Smeter;
      Time_smeter = millis();
    }
  }

//---------- freqency EEP auto write
  if(Flg_eepWT == 1){                       // EEPROM auto Write
    if(Time_Passd+2000 < millis()){
      eep_wdata();
      Flg_eepWT = 0;
    }
  }
}

//----------  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);
}

//----------  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;
}

//----------  EEPROM Dat Read  -----------------------------------------

void eep_rdata(){
  Vfo_Dat = eep_read4(Eep_Freq);
  Enc_Step = eep_read4(Eep_Step);
  Flg_Oflg = EEPROM.read(Eep_Oflg);
  Oft_Dat = eep_read4(Eep_Odat);
}

//----------  EEPROM Dat Write  ----------------------------------------

void eep_wdata(){
  eep_write4(Vfo_Dat,Eep_Freq);
  eep_write4(Enc_Step,Eep_Step);
}  
   
//----------  EEPROM Initialization ------------------------------------

void eep_init(){
  int i;

  for (i=0;i<128;i++)                       // 0 clear(128byte)
    EEPROM.write(i, 0);

  eep_write4(DEF_F,Eep_Freq);
  eep_write4(STEP_1k,Eep_Step);            // Step(1kHz)
  EEPROM.write(Eep_Oflg,Flg_Oflg);         // Offset flag(0)  
  eep_write4(Oft_Dat,Eep_Odat);

  EEPROM.write(Eep_Init,Int_End);           // Init end set(73)  
}

//----------  Rotaly Encorder  -----------------------------------------

void rotary_encoder() {                     // rotary encoder events
  uint8_t 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();
    }
  }
}

//----------  Encorede STEP  -------------------------------------------

void enc_step() {
  if(Enc_Step == STEP_10k)                  // 10k?
    Enc_Step = STEP_100;                    //  yes,100Hz set
  else    
    Enc_Step = Enc_Step * 10;              
}

//----------  STEP Display  --------------------------------------------

void step_disp() {
  oled.setCursor(109, 3);
  oled.setFont(&Tlabels);
  if (Enc_Step == STEP_100)
    oled.print("7");                        // 100
  else if(Enc_Step == STEP_1k){
    oled.print("8");                        // 1k
  }                                        
  else{
    oled.print("9");                        // 10k
  }                                        
}

//---------- Frequency renge over --------------------------------------

void fover_disp() {
  oled.setFont(&Tlcdnums14x24);
  oled.setCursor(1, 0);
  oled.print("--.---.--");
}

//----------  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 ' ' in lcdnums14x24.h
  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);
}

//---------- TX display ------------------------------------------------

void tx_disp() {
  oled.setCursor(2, 3);
  oled.setFont(&Tlabels);
  oled.print("1");                          // TX
}

//---------- Mode display ----------------------------------------------

void oft_disp(byte mod) {
  oled.setCursor(2, 3);
  oled.setFont(&Tlabels);
  if(mod == 0)
    oled.print(".");                      // "." is "   " in Tlabels.h
  else if(mod == 1)
    oled.print("3");                      // "3" is "LSB"
  else if(mod == 2)
    oled.print("4");                      // "4" is "USB"
}

//----------  S-Meter Display  -----------------------------------------

void sm_disp() {
  uint8_t a = 0;
  uint8_t m = 0;

  Val_Smeter = map(Val_Smeter,0,225,0,1023);  
  a = Val_Smeter  / 113;               // 1023 / 9 characters for S = 1,3,5,7,8,9,+10,+20,+30
  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
}

//----------  band chek  -----------------------------------------------

void band_check(unsigned long freq){
  if((freq < LOW_F) || (freq > HI_F))
    Flg_Over = 1;
  else
    Flg_Over = 0;
}

ダウンロード

回路図、スケッチなどは、JA2GQP’sDownload siteの6m AM TRXフォルダーからダウンロード可能。

  

2024年12月4日水曜日

6m AM TRX

 
受信部は、CD2003を使ったダブルスーパー方式。また送信部は、LM386を使ったFCZ方式のトランスレス変調で、10mW程のAM変調された信号をMHW6342で増幅して2W程の出力を得ている。MHW6342は出力3W位の能力しかないので、AMで使った場合2W得るには実力不足である。その為、無変調時に出来る限り深く変調がかかる様にDC-DCコンバータの電圧調整をして小声で話しても出力変動が小さくなる様調整する。実際のオペレーションでは小声で一定音量が望まれる。送信部電源は、ドライバー段にDC-DC降圧コンバータ(LM2596S)と終段にDC-DC昇圧コンバータ(B628)を使って細かな電圧調整が出来るようにした。VFOは、チェック用VFOを使って送受確認を行った。最終版VFOは別途公開予定。

回路図

ダウンロード

回路図などJA2GQP's Download siteの6m AM TRXフォルダからダウンロードできる。









     

2023年10月18日水曜日

GQP1614

 

SOP14 ATtiny1614にレギュレータとUPDI端子を含めたボードを作った。これにより、Arduino nanoライクなボードになった。






裏面にレギュレータ回路がある。今迄、3.3Vの時Aduino miniを使っていたが、ATtiny1614を使うと3.3Vでも16MHzで動作させることが出来るので頼もしい。必要に応じ、3.3Vまたは5Vのレギュレータを実装できる。







回路図



















ダウンロード
JA2GQP's Dwonload siteのGQP1614フォルダから回路図、garver fileをダウンロード出来る。


2023年9月26日火曜日

oled_1614VFO

ATtiny1614機能チェックを目的に、oled mono band VFOを作った。デジタル入力、アナログ入力、I2C、割込み、EEPROM と一通りのチップ機能が確認できた。VFOとしても満足できる仕様だと自負してる。


回路図

ブレッドボードでは、試験に必要な最小限になっている。
 












スケッチ

///////////////////////////////////////////////////////////////////////
//  si5351a oled(128x32) ATtiny1614 VFO program ver.1.0
//    Copyright(C)2023.JA2GQP.All rights reserved.
//
//                                                2023/9/24
//                                                  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
////////////////////////////////
Rotary r = Rotary(2, 5);

////////////////////////////////
// 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 SW_RIT  = 4;               // RIT SW
const byte SW_STEP = 1;               // STEP SW
const byte SW_MODE = 3;               // MODE SW
const byte SW_TX = 0;                 // TX/RX
const byte AD_SM = 10;                // S-meter AD

//---- 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 = 49;             // 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
char Enc_Dir = 0;                     // -1 DIR_CCW, 0 DIR_NONE, 1 DIR_CCW
unsigned int Val_Smeter = 0;
unsigned int Val_Smeterb = 1;

unsigned int Time_Passd;              
byte Flg_eepWT = 0;                   // EEP Write Flag

byte Byt_Mode = LSB;
byte Flg_Over;
byte Flg_Tx = 0;
byte Flg_Rit = 0;                     // RIT Flag

//----------  setup  ----------------------------------------------------

void setup() {
  pinMode(SW_STEP, INPUT_PULLUP);
  pinMode(SW_TX, INPUT_PULLUP);          
  pinMode(SW_RIT, INPUT_PULLUP);          
  pinMode(SW_MODE, INPUT_PULLUP);          

  attachInterrupt(2,rotary_encoder,CHANGE);
  attachInterrupt(5,rotary_encoder,CHANGE);

  oled.begin();
  oled.clear();
  oled.on();

  if(EEPROM.read(Eep_Init) != Int_End){    
    delay(10);
    eep_init();
  }
  else{
    if(digitalRead(SW_STEP) == LOW){
      delay(10);
      eep_init();
      fover_disp();
      while (digitalRead(SW_STEP) == LOW);
    }
  }
  eep_rdata();
  step_disp(Enc_Step);
}

//----------  main(Loop)  -----------------------------------------------

void loop() {

  //---------- receive
  if(digitalRead(SW_TX) == HIGH){        
    Flg_Tx = 0;
    si5351a_enable(0x00);                  // si5351 output enable

    //----------  RIT proc
    if(Flg_Rit == 1){
      Rit_Dat += Enc_Dir * Enc_Step;
      Enc_Dir = 0;
      Rit_Dat = constrain(Rit_Dat,LW_RIT,HI_RIT);
      si5351aSetFrequency(Vfo_Dat + IF_FREQ + Rit_Dat);
      rit_disp(Rit_Dat);
    }

    //---------- normal proc
    else{
      Vfo_Dat += Enc_Dir * Enc_Step;
      Enc_Dir = 0;
      band_check();                        // range check
      si5351aSetFrequency(Vfo_Dat + IF_FREQ);
      freq_disp(Vfo_Dat);                  // frequency display
    }

    //---------- STEP SW check
    if (digitalRead(SW_STEP) == LOW) {    
      enc_step();
      step_disp(Enc_Step);
      while (digitalRead(SW_STEP) == LOW);
    }

    //---------- MODE SW check
    if (digitalRead(SW_MODE) == LOW) {    
      mode_set();
      mode_disp(Flg_Tx,Byt_Mode);
      while (digitalRead(SW_MODE) == LOW);
    }

    //---------- RIT SW check
    if (digitalRead(SW_RIT) == LOW){    
      if(Flg_Rit == 0)
        Flg_Rit = 1;
      else{
        Flg_Rit = 0;
        Rit_Dat = 0;
      }
      while (digitalRead(SW_RIT) == LOW);
    }

  }

  //---------- 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;
    }
  }
}

//----------  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;
}

//----------  Rotaly Encorder  -----------------------------------------

void rotary_encoder() {                     // rotary encoder events
  uint8_t result = r.process();

  if(Flg_Tx == 0){
    if (result) {
      if (result == DIR_CW)
        Enc_Dir = 1;
      else
        Enc_Dir = -1;
    }
  }
  Flg_eepWT = 1;
  Time_Passd = millis();
}

//----------  S-Meter Display  -----------------------------------------

void sm_disp() {
  uint8_t a = 0;
  uint8_t m = 0;

  a = (Val_Smeter + 3) / 113;  // 1024 / 9 characters for S = 1,3,5,7,8,9,
                 // +10,+20,+30
  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);
}


Download

必要なファイルは、Download siteのATtiny1614フォルダからダウンロード可能。