アフィリエイト広告を利用しています
Ad×Ad


Ad×Adは表示されるだけで報酬がもらえます。
以下から登録すると100ptもらえます。
 → アドアド -あなたの街の無料広告サイト-
検索
最新記事

広告

この広告は30日以上更新がないブログに表示されております。
新規記事の投稿を行うことで、非表示にすることが可能です。
posted by fanblog

2017年12月18日

modeequiv() について調べたこと(備忘録)

ソースコードは下記の通り。


void
modeequiv()
{
   static const struct eqdic_s {
       byte   tkn;
       byte   tdif;
       int    rnum;
       word   wu;
   } eqdic[] = {
       {20,  2,  52,  4000},
       {10,  4, 104,  4000},
       { 4, 10, 260, 10000},
       { 2, 20, 520, 20000},
   };
   const struct eqdic_s   *eq;
   int    realnum, i;
   byte   at, crcnt, tokadif, toka, tokanum;
   byte   ch, chnum, vh, adch, adchT;
   word   ui1, waituntil, sinterval;

   rminit(false);

   eq = &eqdic[oscspeed & 3];
   tokanum   = eq->tkn;
   waituntil = eq->wu;
   realnum   = eq->rnum;
   tokadif   = eq->tdif;
   sinterval = 40;      // 20us

   uartjob();

   // ADMUX reg values
   switch(oscinput) {
   default:
   case 0x00: adch = 0x61; chnum = 1;  break;
   case 0x01: adch = 0x62; chnum = 1;  break;
   case 0x02: adch = 0x61; chnum = 2;
       tokanum >>= 1;
       tokadif <<= 1;
       break;
   }
   adchT = 0x61 + (osctrig & 7);

   header(2, 0);
   // This data packet contines to MARC-A

   sinterval--;
   crcnt = 0;
   at = 0;
   for(toka = 0; toka < tokanum; toka++) {
       for(ch = 0; ch < chnum; ch++) {
           // reset and initialize timer1
           TCCR1B = 0x00; // stop, set normal mode
           TCCR1A = 0x00;
           TIMSK1 = 0x00; // no irq
           ICR1   = 0x0000;
           TCNT1  = 0x0000;
           TIFR1  = 0x27; // clear flags;

           // analog comparator setting
           // The D6 pin is the positive input.
           // The negative input is A1, A2, A3, or A4 pin.
           ACSR   = 0x94;  // analog comparator off
           DIDR1  = 0x03;  // disable the digital input func of D6 and D7.
           ADMUX  = adchT; // select the negative input
           ADCSRA = 0x04;
           ADCSRB = 0x40;

           // start time1 with pre=1/8 (2MHz)
           // input capture noise canceler ON
           TCCR1B = (osctrig & 0x10) ? 0xc2 : 0x82;  // edge selection
           ACSR   = 0x14; // capture-on, aco to caputure timer1
           TIFR1  = 0x27; // clear flags again

           ui1 = (tokadif * toka) + (osctdly << 1);

           // falling edge detection(rising edge for ICES1)
           // doesn't stabilize without a 20usec wait below.
           while(TCNT1 < 40)
               ;
           TIFR1 = 0x27;
           // wait until a trigger event happens
           while(true) {
               if (TIFR1 & 0x20) {
                   // trigger event has happened.
                   ui1 += ICR1;
                   at = 0; // a trigger event has happened.
                   break;
               }
               if (TCNT1 > waituntil) {
                   ui1 += TCNT1;
                   at = 1; // trigger failed.
                   break;
               }
               uartjob();
           }

           // at:0 -> trigger event has happened, 1 -> not happened
           ACSR   = 0x94; // disable analog comparator
           ADCSRB = 0x00;
           ADCSRA = 0x84; // adc enable

           TCCR1B = 0x1a; // timer1 CTC-ICR1 mode pre1/8
           TCCR1A = 0x00; //             CTC mode;
           ICR1   = ui1;
           TIFR1  = 0x27; // clear flags

           ADMUX  = 0x60; // adc target is A0 pin to get trigger value;
           ADCSRB = 0x07; // timer1 capture event;
           ADCSRA = 0xf4; // adc auto trigger, force 1st conversion

           // wait until the 1st conversion finishes.
           while((ADCSRA & 0x10) == 0x00)
               uartjob();
           vh = ADCH;  // trigger level
           osctvolt = vh;

           ADMUX = adch + ch;
           ADCSRA = 0xb4;   // clear flag, 1MHz, adate on

           if (toka == 0 && ch == 0) {   // needed only for the 1st loop
               // MARC-A  continued
               txput1(at);
               txput1(vh);
               txputcrc(false);
               txgo(); // start to trasmit a packet
               if (at)
                   goto  ex;  // send header only when trigger failed
           }

           for(i = 0; i < realnum; i++) {
               while(true) {
                   if (TIFR1 & 0x20) {
                       ICR1 = sinterval;
                       TIFR1 = 0x27; // clear timer1 flags;
                   }
                   if ((ADCSRA & 0x10) != 0x00)
                       break;
                   uartjob();
               }
               vh = ADCH;
               ADCSRA = 0xb4;   // clear flag, 1MHz, adate on
               txput1(vh);

               if (++crcnt >= 200) {
                   crcnt = 0;
                   // cause crc error on purpose if trigger failed(at > 0).
                   txputcrc((at > 0) ? true : false);
               }
           }
       }
   }
   //if (crcnt > 0)
   //    sysdown(800);
   if (crcnt == 40) {
       txputcrc((at > 0) ? true : false);
   }
   else
       sysdown(800);


ex:
   txfinish(true, true);
}

2017年12月17日

txputには、txput0(byte ch), txput1(byte ch), txputcrc(boolean force_error) がある

ソースコードは下記の通り。


void
txput0(byte ch)
{
   // to reduce the cpu consumption,
   // venture to omit txn overflow check.
   // if programmed properly, such an overflow never occurs.
   // rxn(appears later) check is omitted as well.
   txbuf[txn++] = ch;
   txcrc = 0;
}


void
txput1(byte ch)
{
   txbuf[txn++] = ch;
   txcrc = crctbl[txcrc ^ ch];
}


void
txputcrc(boolean force_error)
{
   txput0((force_error) ? ++txcrc : txcrc);
}

uartjob() について調べたこと(備忘録)

ソースコードは下記になります。

rollmodeだとrmbuf[]にADCH(ADCの上位8bit)を書き込む。
そうじゃないと、UDR0にtxbuf[]を書き込む。


void
uartjob()
{
   byte    sts;//sts は、status のことらしい

   sts = UCSR0A;//UCSRnA - USARTn制御/状態レジスタA (USART Control and Status Register n A)
   if ((char)sts < 0)  //UCSR0Aの第7bitが1、つまり受信バッファに未読データありということ
       rxbuf[rxn++] = UDR0;//UDRn - USARTnデータ レジスタ (USART I/O Data Register n)
                       //受信バッファにデータがあるのでrxbuf[]に取り込んだ
   // in case rxbuf[] overflow, no fatal situation happens.
   // because rxn is an 8 bit variable and rxbuf[] size is 256.

   if (rmon) {//rmonって何? rmってroll memoryのことらしい
              //rmonは、onがtrueの時に1としているみたい、falseなら0
              //ここは殆ど通らないところ?
       if (TIFR1 & 0x20)//TIFR1 - タイマ/カウンタ1割り込み要求フラグ レジスタ (Timer/Counter 1 Interrupt Flag Register)
                        //カウンタ1がTOP値になった
           TIFR1  = 0x27;  // clear timer1 flags; タイマ/カウンタ1割り込み要求フラグを全部クリアしている
       if (ADCSRA & 0x10) {//ADCSRA - A/D制御/状態レジスタA (ADC Control and Status Register A)
           if (rmon == 1) {
               ICR1  = 100 - 1;      // 50us
                   //ICR1は16bitレジスタ カウンタに設定する値
                   //0.5usを100倍すると50us
               ADMUX = 0x62;//ADMUX - A/D多重器選択レジスタ (ADC Multiplexer Select Register)
                   //0x62 基準電圧はAVCCでアナログ入力はADC2 PC2 下記のコメントと矛盾
               rmon = 2;
               rmbuf[rmw++] = ADCH;  // CH1(A1pin) value
                   //ADCの値は、8bit精度ならADCHを読むだけで事足りる(ADCLは読まない)
           }
           else if (rmon == 2) {
               ICR1  = 400 - 1;      // 200us
                   //0.5usを400倍すると200us
               ADMUX = 0x60;//0x60 基準電圧はAVCCでアナログ入力はADC0 PC0 下記のコメントと矛盾
               rmbuf[rmw++] = ADCH;  // CH2(A2pin) value
               rmon = 3;
           }
           else {
               ICR1  = 500 - 1;      // 250us
                    //0.5usを500倍すると250us
               ADMUX = 0x61;//0x61 基準電圧はAVCCでアナログ入力はADC1 PC1 下記のコメントと矛盾
               rmon = 1;
               osctvolt = ADCH;       // trigger level
           }
           ADCSRA = 0xb4;  // clear flags, 1MHz, adate on
               //ADCSRA - A/D制御/状態レジスタA (ADC Control and Status Register A)
               //システムクロック16分周
           return; //in order to release cpu quickly
       }
   }

   if (txr < txn && (sts & 0x20)) {
       UCSR0A = (sts & 0xe3) | 0x40;
       UDR0 = txbuf[txr++];//UDRn - USARTnデータ レジスタ (USART I/O Data Register n)
           //UDR0は送受信バッファ、バッファに送信データを書き込む
   }
}

sysdown(int dly) について調べたこと(備忘録)

ソースコードは下記の通り。

レジスターに値を代入し、D13のLEDを点滅させる内容。


void
sysdown(int dly)  // dly .. in msec
{
   int    i;
   byte   s;

   SPCR   = 0x00; // disable SPI
   TCCR0B = 0x00; // stop timer0
   TCCR0A = 0x02; // ctc mode
   TIMSK0 = 0x00; // Disable all timer0 irqs
   TIFR0  = 0x07; // clear flags
   TCNT0  = 0x00;
   OCR0A  = 250 - 1;
   TCCR0B = 0x03; // start timer0  pre = 1/64  i.e 250kHz

   s = 0;
   while(true) {
       if (++s & 1)
           PORTB |= 0x20;  // D13 == HIGH  (LED on)
       else
           PORTB &= 0xdf;  // D13 == LOW   (LED off)
       for(i = 0; i < dly; i++) {
           while((TIFR0 & 2) == 0)
               ;
           TIFR0  = 0x07; // clear flags
       }
   }
}



2017年12月15日

txinit()について調べたこと(備忘録)

ソースコードは下記の通り。


void
txinit(void)
{
   if (txn >= TXBSZ)      // TXBSZは1100のこと、ソースコードの冒頭に定義してありました!
       sysdown(200);      // tx buffer overflow has been detected.
   txn = 0;
   txr = TXBSZ;
}

header(byte typ, byte reqid)について調べた(備忘録)

ソースコードは下記の通り。ヘッダーのデータを作成しているらしい。

詳細は現時点では不明。
@Arduino_UNOに送るためのオシロスコープの設定を作っているらしい。
Atxbuf[]に書き込む回数と読みだす回数が一致していない理由が不明。


void
header(byte typ, byte reqid)
{
   static byte seq;//staticだから前の値を保持している

   txinit();
   uartjob();

   // prologue
   //序章?
   txput0(0xaa);//txbuf[]に書き込み
   txput0(0x55);//txbuf[]に書き込み
   txput0(0xa5);//txbuf[]に書き込み
   txput0(0x5a);//txbuf[]に書き込み

   uartjob();//ここまでtxbuf[]に4回書き込んだが、txbuf[]から読みだしたのはこの1回、

   txput1(typ);
   if (typ == 3)//3だとrollmodeなの?
       txput1(seq++);  // rollmode
   else
       txput1(reqid);
   txput1(oscspeed);
   txput1(oscinput);
   uartjob();//txbuf[]に書き込む回数と読みだす回数が一致しないのは?

   txput1(osctrig);
   txput1(osccupgain);
   txput1(osctdly >> 8);
   txput1(osctdly);
   uartjob();

   txput1(oscofreq >> 16);
   txput1(oscofreq >> 8);
   txput1(oscofreq);
   txput1(oscoduty);
   uartjob();
}

moderoll()について調べたこと(備忘録)

moderoll()は、kit_scope.ino 中の loop() の中にある関数です。

詳細はまだ不明。

@rmon==0の時(プログラム開始直後の時か?)
 各種レジスターに値を代入

Aheader(3, 0)を実行
 txput1(0)
 txput1(osctvolt)
 txputcrc(false)
 上記は何かしらデータを送信している

Btxgo()を実行 パケットを送信開始

Cfor文を実行 この中でおそらく初めにヘッダーデータを、次に波形データを送信しているらしい

Dtxputcrc(false)
 txfinish(true, true) を実行し、一連のデータ送信が完了ということらしい

ソースコードは下記の通り。


void
moderoll()
{
   byte    i;

   if (rmon == 0) {                         // start rollmode
       // reset and initialize timer1
       TCCR1B = 0x00;   // stop
       TCCR1A = 0x00;
       TIMSK1 = 0x00;   // no irq
       TCNT1  = 0x0000;
       ICR1   = 200;    // 100 usec
       TIFR1  = 0x27;   // clear flags;

       ACSR   = 0x94; // disable analog comparator
       ADCSRB = 0x00;
       ADCSRA = 0x84; // adc enable
       ADMUX  = 0x60; // adc target is A0 pin to get trigger value;
       ADCSRB = 0x07; // timer1 capture event;
       ADCSRA = 0xf4; // adc auto trigger, force start 1st conversion

       TCCR1B = 0x1a; // timer1 CTC-ICR1 mode pre1/8
       TCCR1A = 0x00; //             CTC mode;

       // wait until the 1st conversion finishes.
       while((ADCSRA & 0x10) == 0x00)
           uartjob();
       osctvolt = ADCH;  // trigger level

       ADMUX  = 0x61;
       ADCSRA = 0xb4;   // clear flag, 1MHz, adate on

       rminit(true);
   }

   header(3, 0);

   txput1(0);
   txput1(osctvolt);
   txputcrc(false);

   txgo(); // start to trasmit a packet

   for(i = 0; i < 200; i++) {
       while(rmw == rmr)
           uartjob();
       txput1(rmbuf[rmr++]);
   }
   txputcrc(false);

   txfinish(true, true);
}




kit_scope.ino について調べたこと(備忘録)

kit_scope.inoは、Arduino開発環境のソースコードです。v0.72 Mar 20, 2016版を使用しています。

ATmega328Pのデータシートは支障がない限り、下記(日本語)を参照します。
https://avr.jp/user/DS/PDF/mega328P.pdf

Arduino開発環境については支障がない限り、下記(日本語)を参照します。
http://www.musashinodenpa.com/arduino/ref/

この部分で行っているのは、入力信号のAD変換とUARTでPCへの送信、UARTでPCからのコマンドの受信となります。

Arduino のソースコードの形式はC言語に従いますが、setup()に初期設定を書き、loop()に通常処理を書きます。setup()は文字通り1回だけ実行されます。loop()も文字通りで、これは無限ループになります。setup()とloop()を置く順番は、特に無いようです。大文字で記載されている変数は、CPU(AtMega)のレジスターとなります。レジスターはAtMegaの仕様書に記載されている名称そのままで、変数宣言しないで使用します。cli()の関数もArdino環境固有のもので、全ての割り込みを禁止にする関数です。

オシロスコーププログラムについて調べたこと(備忘録)

九州工業大学情報工学部においてあるオシロスコーププログラムについて調べたことを書いていきます。
Kyutech Arduino Scope
http://www.iizuka.kyutech.ac.jp/faculty/physicalcomputing/pc_kitscope/

オシロスコープ(Kyutech Arduino Scope)のハードウエアはArduino UNOで、ソースコードはArduino環境で作成されているプログラムです。オシロスコープの表示部は、PC側となります。こちらはProcessingで作成されています。

Arduino UNOに、ブレッドボードで簡単な回路を追加するだけで、オシロスコープになるところがとても素晴らしい作品です。

×

この広告は30日以上新しい記事の更新がないブログに表示されております。