連載
FPGAの内蔵温度センサーから値を得る:MAX 10 FPGAで学ぶFPGA開発入門(13)(5/6 ページ)
アルテラのFPGA「MAX 10」には温度センサーが内蔵されており、自身の温度を測定可能だ。メガファンクション「ALTPLL」を使い、内蔵センサーからの値を得るまでを試みる。
Eclipseでのプログラミング
次はEclipseを利用してのプログラミングである。こちらも従来と同じく“File” → “New” → “Nios II Application and BSP from Template”でHello Worldを指定してダミーのソースを生成、それを書き換える形となる。書き換えたソースが以下のList3となる。基本は前回のソースを流用している。
仕様上の変更点としては以下の2点となる。LEDへの表示などはそのままで、選択したほうの温度を表示するようになっている。
- ADC経由でのTSDでの温度取り込みに対応
- これに対応してスイッチをOn/OffからADC経由温度/I2C経由温度/Offの3種類に変更
/* * "Hello World" example. * * This example prints 'Hello from Nios II' to the STDOUT stream. It runs on * the Nios II 'standard', 'full_featured', 'fast', and 'low_cost' example * designs. It runs with or without the MicroC/OS-II RTOS and requires a STDOUT * device in your system's hardware. * The memory footprint of this hosted application is ~69 kbytes by default * using the standard reference design. * * For a reduced footprint version of this template, and an explanation of how * to reduce the memory footprint for a given application, see the * "small_hello_world" template. * */ #include <stdio.h> #include "system.h" #include "altera_avalon_pio_regs.h" #include "altera_modular_adc.h" #include "altera_modular_adc_sequencer_regs.h" #include "terasic_includes.h" #include "I2C.h" #define R_TEMPERATURE 0x00 #define R_HUMIDITY 0x01 #define R_CONFIGURATION 0x02 #define R_MANUFACTURER_ID 0xFE #define R_DEVICE_ID 0xFF #define HDC1000_I2C_ADRESS 0x80 // 1000_000 + Write bit 0 long HEXtable[11] = {0x40, // 0:1000000 0x79, // 1:1111001 0x24, // 2:0100100 0x30, // 3:0110000 0x19, // 4:0011001 0x12, // 5:0010010 0x02, // 6:0000010 0x78, // 7:1111000 0x00, // 8:0000000 0x10, // 9:0010000 0x7F // Blank }; #define KEY_ONDIE 0x01 #define KEY_I2C 0x02 #define KEY_STOP 0x04 void LEDR_out(int num) { if(num > 9) return; if(num < 0) { IOWR_ALTERA_AVALON_PIO_DATA(LEDR_BASE, 0x0); // -の値がきたらBlankにする } else { IOWR_ALTERA_AVALON_PIO_DATA(LEDR_BASE, 1 << num); } } void HEX_out(int digit, int num) { if(num > 9) return; if(num < 0) num=10; // -の値がきたらBlankにする if((digit < 0)||(digit > 2)) return; if(digit) IOWR_ALTERA_AVALON_PIO_DATA(HEX1_BASE, HEXtable[num]); else IOWR_ALTERA_AVALON_PIO_DATA(HEX0_BASE, HEXtable[num]); } bool Read_Configuration(void) { bool bPass; alt_u16 data; bPass = I2C_Read16(RH_TEMP_I2C_SCL_BASE,RH_TEMP_I2C_SDA_BASE,HDC1000_I2C_ADRESS,R_CONFIGURATION,&data); if(bPass){ printf("Read Configuration Data = 0x%x \r\n",data); if(((data>>12)&0x0001) == 0) printf("Acquisition mode:Temperature or Humidity is acquired.\r\n "); else printf("Acquisition mode:Temperature and Humidity are acquired in sequence,Temperature first.\r\n "); if(((data>>10)&0x0001) == 0) printf("Temperature Measurement Resolution: 14 bit.\r\n "); else printf("Temperature Measurement Resolution: 11 bit.\r\n "); if(((data>> 8)&0x0003) == 0) printf("Humidity Measurement Resolution: 14 bit.\r\n "); else if(((data>> 8)&0x0003) == 1) printf("Humidity Measurement Resolution: 11 bit.\r\n "); else printf("Humidity Measurement Resolution: 8 bit.\r\n "); } return bPass; } bool Write_Configuration(alt_u8 acquisition_mode , alt_u8 temp_resolution,alt_u8 humidity_resolution) { bool bPass; alt_u16 data; data = ((acquisition_mode&0x0001) <<12) + ((temp_resolution&0x0001)<<10) + ((humidity_resolution&0x0003)<<8); //printf("Write Configuration Data = 0x%x \r\n",data); bPass = I2C_Write16(RH_TEMP_I2C_SCL_BASE,RH_TEMP_I2C_SDA_BASE,HDC1000_I2C_ADRESS,R_CONFIGURATION,data); return bPass; } bool RH_Temp_Sensor_init(void) { bool bPass; alt_u16 data; bPass = I2C_Read16(RH_TEMP_I2C_SCL_BASE,RH_TEMP_I2C_SDA_BASE,HDC1000_I2C_ADRESS,R_MANUFACTURER_ID,&data); if(bPass){ printf("Manufacturer ID is %x\r\n",data);// 0x5449 expected }else{ printf("I2C MultipleRead Failed!\r\n"); } bPass = I2C_Read16(RH_TEMP_I2C_SCL_BASE,RH_TEMP_I2C_SDA_BASE,HDC1000_I2C_ADRESS,R_DEVICE_ID,&data); if(bPass){ printf("Device ID is %x\r\n",data); // 0x1000 expected }else{ printf("I2C Read16 Failed!\r\n"); } bPass = Read_Configuration(); if(!bPass){ printf("I2C Read_Configuration Failed!\r\n"); } bPass = Write_Configuration(1,0,0); if(!bPass){ printf("[Note] Set Configuration Failed!\r\n"); } return bPass; } int main() { alt_u8 KeyStatus, FLAG=0; // 0:Stop 1:Running bool bPass; alt_u8 rh_temp[4]; alt_u16 data; int temp_sum; int temp_adc; int temp_int; int lpCnt; // Initial Display HEX_out(0, -1); HEX_out(1, -1); LEDR_out(-1); // Sensor Initialize RH_Temp_Sensor_init(); // ADC Initialize and Start ADC sequencer IOWR(ADC_SEQUENCER_CSR_BASE, 0, 0); // Stop ADC usleep(1000); IOWR(ADC_SAMPLE_STORE_CSR_BASE, 64, 0); // Disable Interrupt IOWR(ADC_SEQUENCER_CSR_BASE, 0, 1); // Enable ADC while(1) { // Wait 0.5sec usleep(500000); // Examine KEY_STOP & KEY_I2C/KEY_ONDIE KeyStatus = IORD_ALTERA_AVALON_PIO_DATA(KEY_BASE); if(( KeyStatus & KEY_STOP ) == 0 ) { FLAG = 0; HEX_out(0, -1); HEX_out(1, -1); LEDR_out(-1); } else if (( KeyStatus & KEY_ONDIE ) == 0 ) { FLAG = 1; } else if (( KeyStatus & KEY_I2C ) == 0 ) { FLAG = 2; } switch( FLAG ) { case 1: // Read Temperature from ADC for(lpCnt=0, temp_sum=0; lpCnt<64; lpCnt++) { temp_adc=IORD(ADC_SAMPLE_STORE_CSR_BASE,lpCnt); temp_sum+=temp_adc; } temp_sum /= 64; temp = (double)temp_sum*(double)temp_sum*-0.0003054+(double)temp_sum*1.7626-2325; break; case 2: // Read Temperature via I2C bPass = I2C_MultipleRead(RH_TEMP_I2C_SCL_BASE, RH_TEMP_I2C_SDA_BASE, HDC1000_I2C_ADRESS, R_TEMPERATURE, rh_temp,4); if(bPass) { data = (rh_temp[0] << 8) | rh_temp[1] ; data = (rh_temp[0] << 8) | rh_temp[1] ; temp = (double)data/397.188 -40.0; } else { FLAG = 0; // Failed to read via I2C } } if ( FLAG ) { temp_int = (int)(temp*10.0); HEX_out(1,temp_int/100); HEX_out(0,(temp_int%100)/10); LEDR_out(temp_int%10); printf("Temperature: %.1f \n",temp); } } return 0; }
List3
Copyright © ITmedia, Inc. All Rights Reserved.
関連記事
- オンボードされた温湿度センサーからFPGAで値を得る
FPGA「MAX 10」搭載開発ボード「MAX 10 NEEK」には各周辺機器が備えられており、そこにはTI製の温湿度センサーも含まれる。サンプルプログラムを元に、オンボードされたデバイスの制御を試みる。 - 「MAX 10 NEEK」でストップウォッチを開発し、内蔵メモリから起動する
アルテラのFPGA「MAX 10」を搭載した開発ボード「MAX 10 NEEK」にはLEDやフラッシュメモリなどの各周辺機器が備えられている。今回はLEDを使ったストップウォッチを開発し、内蔵メモリから起動する。 - 「MAX 10 NEEK」に搭載されたDDR3メモリを使う
MAX10搭載開発ボード「MAX 10 NEEK」には各周辺機器が備えられており、その中にはDDR3メモリも含まれる。ソフトコアCPU「Nios II」からの利用も含めて手順を紹介する。 - 「MAX 10 NEEK」へソフトコアCPUを組み込む
周辺機器の充実したMAX10搭載開発ボード「MAX 10 NEEK」に、ソフトコアCPU「NIOS II」を組み込み、ソフトコアCPUからボード搭載LEDの制御までを紹介する。 - 周辺機器の充実した「MAX 10 NEEK」で本格的な開発を目指す
FPGA開発において周辺機器(回路)は見落とせない要素の1つ。タッチパネルや各種出入力などを備えた「MAX 10 NEEK」を導入して、本格的な開発を目指す。