この広告は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[]を書き込む。
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に、ブレッドボードで簡単な回路を追加するだけで、オシロスコープになるところがとても素晴らしい作品です。