/************************************************************************************ Side Tone Osc By Compare Mode v1.1 2015/9 By JA3OOK T.nakamura PIC Capture/Compare/PWM モジュールの Compare Mode を使用することにより、     パルス幅変調信号を発生し、MCUの外につけたRCフィルターで波形を整えている。 ・パルスの開始:タイマー割込み時点 ・パルスの終了:CCP割込み時点 ・タイマー割込み毎にCCPR値(パルス幅設定レジスタ値)を周期的に変化させ、      CCP割込みタイミングを連続的に変更し、パルス幅を連続的に変更させオーディオを生成   参考 1 MCUのクロック周波数が高いほど、よりきれいな正弦波を発生できる。 2 LCDは無くても使用には差し支えない。あれば開発やプログラム変更に便利。     ------------------------------------------------------------------------------------- 開発環境 MPLAB X IDL ver 3.05 MPLAB XC8 C Compiler Free mode v1.34 CPU: PIC16F1936 1:MCLR:1 2:RA0:0:LCD ctrl 3:RA1:0:LCD ctrl 4:RA2:0:LCD ctrl 5:RA3:0:LCD ctrl 6:RA4:0:LCD ctrl 7:RA5:0:LCD ctrl 8:VSS 9: RA7: 10:RA6: 11:RC0: 12:RC1: 13:RC2: 14:RC3: 15:RC4: 16:RC5: 17:RC6: 18:RC7: 19:VSS 20:VDD 21:RB0:AN12:1:出力周波数変更指示受け用電圧測定(可変抵抗器) 22:RB1:0:動作開始確認および出力中表示用LED 23:RB2:1:CW-key入力 24:RB3:0: 25:RB4:0: 26:RB5:0: 27:RB6:0:低周波出力(パルス変調出力) 28:RB7:0: *************************************************************************************/ // refer \c:Program Files\Microchip\xc8\v1.34\docs\chips\16f1936.html #pragma config CLKOUTEN =OFF//CLKOUT function is disabled. I/O or oscillator function on the CLKOUT pin #pragma config WDTE =OFF //WDT disabled #pragma config PWRTE =OFF //Power-up Timer disable #pragma config CP =OFF //Code protection off #pragma config BOREN =OFF //BOR disabled #pragma config FCMEN =OFF //Fail-Safe Clock Monitor disabled #pragma config MCLRE =OFF //MCLR/VPP pin function is digital input #pragma config CPD = OFF //Code protection off #pragma config IESO =OFF //Internal External Switchover mode disabled #pragma config FOSC =INTOSC //INTRC oscillator; port I/O function on both RA6/OSC2/CLKO pin and RA7/OSC1/CLKI pin #pragma config PLLEN =OFF //4x PLL disabled #pragma config STVREN =OFF //OFF Stack Overflow or Underflow will not cause a Reset #pragma config LVP =OFF //High-voltage on MCLR/VPP must be used for programming #pragma config WRT =OFF //Write protection off #include //#include //#include #include #include "lcd.h" #define MHz 000000 #define _XTAL_FREQ 32MHz //PIC_OSC_FREQ #define signal_port LATB6 //output #define led_port LATB1 //output #define cw_key RB2 //input internal weak pull-up signed int tm1_count; // タイマーの割込み発生回数をカウントする変数 signed int t1count; // タイマー1用カウント signed int t1countH; // タイマー1用カウント signed int t1countL; // タイマー1用カウント #define pulse_max 36 // 1サイクル当たりのパルス数 #define wave_style_bunbo 128 //wave_styleの分母の値 この値が安全(実験して決定する) //#define wave_style_bunbo 115 //分母の値が115以下、そしてCCPR値が-46以下の //場合、波形が乱れる。 乱れる限界ははっきりせず、115は一例。 //推定:パルス間隔が狭くくデューティ比が高いと乱れると思われる。 unsigned char wave_style_sin[pulse_max] = {50,59,67,75,82,88,93,97,99,100,99,97,93,88,82,75,67,59,50,41,33,25,18,12,7,3,1,0,1,3,7,12,18,25,33,41}; //上記各値はExcelで計算→csvで出力→エディタ→コピーペースト signed int wave_style_ccpr[pulse_max]; //最新のタイマー値でのccpr値を格納 unsigned int ad_val;//AD変換後の値 unsigned int ad_val_old; unsigned int timer; //時間稼ぎ用(符号無し16ビット長) unsigned char i; //汎用 unsigned int i_uint; //汎用 unsigned int j_uint; //汎用 unsigned char hexch[] ="0123456789ABCDEF";//16進数表示変換テーブル unsigned char str[10]; //汎用 主に数値の十進数表示変換に使用 void //intやchar正数をASCII文字に変換し*bufferに格納されるので文字として送信したり液晶に表示できる Bin2str(unsigned int data, char *buffer, unsigned int digit) { // 引数:intやchar正数データ, 文字データ, 期待する文字データの文字数 char i; buffer += digit; // 最後の数字位置 += : Buffer=Buffer+digit *buffer = 0; // 文字列終端(文字数+1の位置)へ0X00 for(i=digit; i>0; i--) { // 変換は下位から上位へ buffer--; // ポインター1 *buffer = (data % 10) + 0x30;// ASCIIへ  % : 割り算の余り data = data / 10; // 次の桁へ } } void //int正数を16進のASCII文字に変換しstrに格納されるので文字として送信したり液晶に表示できる DispHex(unsigned int j){ char i; i = j >> 12; str[0] = hexch[i]; //上位4bit表示 max:03FF i = (j >> 8) & 0x000F; str[1] = hexch[i]; //下位4bit表示 max:03FF i = (j >> 4) & 0x000F; str[2] = hexch[i]; //上位4bit表示 max:03FF i = j & 0x000F; str[3] = hexch[i]; //下位4bit表示 max:03FF str[4] = 0; //set NULL } void interrupt Interrupt_Service_Routine( void ){ // タイマー割込み //GIE = 0 ; //全割込みをマスクする ← これはPICのHWが自動的に行う if (TMR1IF) { // タイマー1の割込み発生か? tm1_count ++; if(tm1_count >= pulse_max) { tm1_count = 0; } CCPR1H = (wave_style_ccpr[tm1_count] >> 8); // 比較値の初期化 CCPR1L = (wave_style_ccpr[tm1_count] & 0x00FF) ; // 比較値の初期化 //割込間隔は t1countH,t1countL が保持している TMR1H = t1countH; // タイマーの初期化 TMR1L = t1countL; // タイマーの初期化 TMR1IF = 0 ; // タイマー割込フラグをリセット signal_port = 1; } if(CCP1IF) { // CCP1 割込み発生か? signal_port = 0; CCP1IF = 0; } } void Analize_voltage(void){ Lcd_goto(0x0D); //debug Bin2str(wave_style_bunbo, str,3); //debug Lcd_puts(str); //debug while(1){ //無限loop if(cw_key == 0){ //CW電鍵押下(ON) TMR1IE = 1 ; // タイマー割込をマスクしない led_port=1; } else { TMR1IE = 0 ; // タイマー割込をマスク led_port=0; } Lcd_goto(0x06); //debug if(t1count < 0){ //debug Lcd_putch('-'); //debug i_uint = 0 - t1count; //debug } //debug else { //debug Lcd_putch('+'); //debug i_uint = t1count; //debug } //debug Bin2str(i_uint, str,5); //debug Lcd_puts(str); //debug Lcd_goto(0x40); //debug if(wave_style_ccpr[9] < 0){ //debug Lcd_putch('-'); //debug i_uint = 0 - wave_style_ccpr[9]; //debug } //debug else { //debug Lcd_putch('+'); //debug i_uint = wave_style_ccpr[9]; //debug } //debug Bin2str(i_uint, str,5); //debug Lcd_puts(str); //debug if(wave_style_ccpr[27] < 0){ //debug Lcd_putch('-'); //debug i_uint = 0 - wave_style_ccpr[27]; //debug } //debug else { //debug Lcd_putch('+'); //debug i_uint = wave_style_ccpr[27]; //debug } //debug Bin2str(i_uint, str,5); //debug Lcd_puts(str); //debug //電圧測定 ADON = 1; //AD変換を開始せよ __delay_ms(5); //ソースプログラム v1.1 追加 //アクイジションAcquisition時間(取得=充電時間) See Data Sheet 165P GO_nDONE = 1; //AD変換開始 while(GO_nDONE); //変換完了待ち ad_val = (ADRESH << 8) | ADRESL; // 0 =< ad_val =<1023 //測定電圧生データの表示 //debug Lcd_goto(0x00); //debug Bin2str(ad_val, str, 4);//debug Lcd_puts(str); //debug //電圧測定終了 /* //実験 波形乱れの限界値テスト //Compare mode用タイマー割込み間隔を t1count にセット // t1count = 0.45 * ad_val - 500 (つまりセットする値の範囲は -500から-40) // 出力周波数は 400 から 2370Hz  MPUにより違いが出るだろう i_uint = ad_val * 45; i_uint = i_uint / 100; t1count = i_uint - 500; t1countH = (t1count >> 8); // タイマーの初期化 t1countL = (t1count & 0x00FF) ; // タイマーの初期化 */ //Compare mode用タイマー割込み間隔を t1count にセット // t1count = 0.33 * ad_val - 500 (つまりセットする値の範囲は -500から-162) // 出力周波数は 400 から 1033Hz  MPUにより違いが出るだろう i_uint = ad_val * 33; i_uint = i_uint / 100; t1count = i_uint - 500; t1countH = (t1count >> 8); // タイマーの初期化 t1countL = (t1count & 0x00FF) ; // タイマーの初期化 //wave_style_ccprを計算 i_uint = 0 - t1count; //cur_ccpr = wave_style[] * t1count / wave_style_bunbo for(i=0; i: ECCPx Mode Select bits //1010 = Compare mode: generate software interrupt on compare match (CCPx pin is unaffected, // CCPxIF is set) CCP1M3 = 1; //タイマー1 CCP ソフトウエア割込み CCP1M2 = 0; //タイマー1 CCP ソフトウエア割込み CCP1M1 = 1; //タイマー1 CCP ソフトウエア割込み CCP1M0 = 0; //タイマー1 CCP ソフトウエア割込み //高い出力周波数検討 // Fosc:32MHz=0.03125μs(Tosc=FOSC*4=0.125μs) Prescale=1に於いて // 1000Hzを得るには // 1000Hz=1000μs 36パルスで1サイクル なので、割込間隔=1000μs/36=27μs // 割込みカウンター値=27μs/0.125μs=216 // 出力周波数を実測すると830Hz  割込みカウンター値170で実測1000Hz //低い出力周波数検討 // Fosc:32MHz=0.03125μs(Tosc=Fosc*4=0.125μs) Prescale=1に於いて // 500Hzを得るには // 500Hz=2000μs 36パルスで1サイクル なので、割込間隔=2000μs/36=55μs // 割込みカウンター値=55μs/0.125μs=440 // 出力周波数を実測すると450Hz  割込みカウンター値400で実測500Hz t1count= -400; //カウントの初期値を仮設定 本設定はAD変換後に頻繁に行う TMR1H = (t1count >> 8); // タイマーの初期化 TMR1L = (t1count & 0x00FF) ; // タイマーの初期化 T1CON = 0b00000000; //プリスケーラカウント値 1:1 内部クロックでTIMERを使用 FOSC/4でTIMER1を使用 TMR1IF = 0 ; // タイマー1割込フラグを0 tm1_count = 0 ; // 割込み発生の回数カウンターを0 Lcd_init(); Lcd_write(0b00001100); //Display on, no cursor Lcd_clear(); /* display open message */ Lcd_goto(0x00); //LCDの表示位置を一行目左端 Lcd_puts("Tone OSC by CPM");//LCDにメッセージを表示 Lcd_goto(0x40); //LCDの表示位置を二行目左端 Lcd_puts("v1.0 ");//LCDにメッセージを表示 Lcd_puts("JA3OOK");//LCDにメッセージを表示 //timer=1; while(timer--){__delay_ms(1000);}; //Wait /* End of hello msg on LCD */ led_port=1; timer=1; while(timer--){__delay_ms(1000);}; //Wait led_port=0; timer=1; while(timer--){__delay_ms(1000);}; //Wait led_port=1; timer=1; while(timer--){__delay_ms(1000);}; //Wait led_port=0; timer=1; while(timer--){__delay_ms(1000);}; //Wait Lcd_clear(); TMR1ON = 1 ; // タイマー開始 TMR1IE = 1 ; // タイマー割込をマスクしない CCP1IE = 1 ; //タイマー1 CCP ソフトウエア割込みをマスクしない PEIE = 1 ; // 周辺割込みをマスクしない GIE = 1 ; // 全割込み処理をマスクしない Analize_voltage(); }