新規記事の投稿を行うことで、非表示にすることが可能です。
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[]を書き込む。
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を点滅させる内容。
レジスターに値を代入し、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[]に書き込む回数と読みだす回数が一致していない理由が不明。
詳細は現時点では不明。
@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) を実行し、一連のデータ送信が完了ということらしい
ソースコードは下記の通り。
詳細はまだ不明。
@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環境固有のもので、全ての割り込みを禁止にする関数です。
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に、ブレッドボードで簡単な回路を追加するだけで、オシロスコープになるところがとても素晴らしい作品です。
Kyutech Arduino Scope
http://www.iizuka.kyutech.ac.jp/faculty/physicalcomputing/pc_kitscope/
オシロスコープ(Kyutech Arduino Scope)のハードウエアはArduino UNOで、ソースコードはArduino環境で作成されているプログラムです。オシロスコープの表示部は、PC側となります。こちらはProcessingで作成されています。
Arduino UNOに、ブレッドボードで簡単な回路を追加するだけで、オシロスコープになるところがとても素晴らしい作品です。