/* ======================================================== MCU CTR BSR 2009/03/30 開発技術部 藤田 ブートローダー部 ホストの通信と、自己書き換え、ファームのチェックを行う。 ======================================================== */ #pragma SFR #pragma di #pragma ei #pragma nop #pragma stop #pragma halt #pragma opc #include "incs_loader.h" #include #include "fsl_user.h" #include "i2c_ctr.h" #include "i2c_mcu.h" #include "pm.h" #include "rtc.h" #include "reboot.h" #include "magic.h" #define REG_BIT_RESF_WDT 0x10 #define REG_BIT_RESF_TRAP 0x80 #define REG_BIT_RESF_LVI 0x01 // ======================================================== #if (FSL_DATA_BUFFER_SIZE>0) fsl_u08 fsl_data_buffer[FSL_DATA_BUFFER_SIZE]; #endif #ifdef FSL_INT_BACKUP static fsl_u08 fsl_MK0L_bak_u08; /* if (interrupt backup required) */ static fsl_u08 fsl_MK0H_bak_u08; /* { */ static fsl_u08 fsl_MK1L_bak_u08; /* reserve space for backup information */ static fsl_u08 fsl_MK1H_bak_u08; /* of interrupt mask flags */ static fsl_u08 fsl_MK2L_bak_u08; /* */ static fsl_u08 fsl_MK2H_bak_u08; /* } */ #endif // ======================================================== void FSL_Open( void ); void FSL_Close( void ); void hdwinit( void ); static void hdwinit2( ); extern void main_loop( ); // ======================================================== void main( ) { while( 1 ) { WDT_Restart( ); if( RTCEN ) // これはTRAP(自己update)にしか使えない。WDT等では立たない { system_status.reboot = true; } { volatile u8 my_resf = RESF; // ←読むと消える if( ( my_resf & REG_BIT_RESF_WDT ) != 0 ) { vreg_ctr[ VREG_C_MCU_STATUS ] |= REG_BIT_STATUS_WDT_RESET; // set_irq( VREG_C_IRQ0, REG_BIT_IRQ_WDT_RESET ); // ↑I2Cの初期化後に行う system_status.reboot = true; } else if( ( my_resf & REG_BIT_RESF_TRAP ) != 0 ) // ( FSL_ForceReset, 'r' ) { system_status.reboot = true; } /* else if(( my_resf & REG_BIT_RESF_LVI ) != 0 ) { vreg_ctr[ VREG_C_MCU_STATUS ] |= ( 1 << 2 ); } */ } if( !system_status.reboot ) { // 通常の電源投入 // 電池接続時、ちょっと待ってみる(チャタリング対策) u16 pwup_delay0; for( pwup_delay0 = 0xFFFF; pwup_delay0 != 0; pwup_delay0-- ){;}; } hdwinit2( ); // ファームの整合性チェック // { u8 i; u8 comp = 0; // ローダーと本体は同じバージョンか? /// 次へのアップデートの途中で終わってないか? for( i = 0; i < sizeof( __TIME__ ); i++ ) // sizeof( __TIME__ ) = 8 らし { comp += ( *( __far u8 * )( MGC_LOAD + i ) == *( u8 * )( MGC_FOOT + i ) ) ? 0 : 1; comp += ( *( u8 * )( MGC_HEAD + i ) == *( u8 * )( MGC_FOOT + i ) ) ? 0 : 1; } if( *( u8 * )( MGC_FOOT +2 ) != ':' ) // 消去済のまま { comp += 1; } if( comp != 0 ) { // ファームリストアを試みる firm_restore( ); // 帰ってこない。リセットをかける。 } } // 試遊台判定 // デバッガつないでると上手に判定できないことがあります。 // P40,41 // HH 実機 // HL NBD試遊台 // LH スタンドアロン試遊台 // LL ISデバッガ if( !DIPSW_0 && !DIPSW_1 ) { system_status.is_dev = 1; } else { #ifndef _FORCE_TAIKENDAI_ if( !DIPSW_0 ) #else if(1) #endif { system_status.taikendai = 1; } else { system_status.taikendai = 0; } #ifndef _FORCE_TAIKENDAI_NBD_ if( !DIPSW_1 ) #else if(1) #endif { system_status.taikendai_nbd = 1; } else { system_status.taikendai_nbd = 0; } } // 通常運転 main_loop( ); } } // ======================================================== void hdwinit( void ) { // スタートアップルーチンが勝手に呼びます DI( ); /* マスタ割り込み禁止 */ CMC = 0b00010000; /* X1発振せず(入力ポート)、XT1使用、超低電力発振は不安定に付き通常で */ CSC = 0b10000000; /* X1発振なし、XT1発振あり、高速内蔵発振動作 */ OSMC = 0x01; /* 隠しレジスタ */ CKC = 0b00001000; /* CPU/周辺クロック=fMAIN、fMAIN=fMX、fCLK=fMX */ /*--- 低電圧検出回路の設定 ---*/ /* リセット解除時のデフォルトは、オプション・バイトにて指定される */ LVIS = 0b00000000; /* VLVI = 4.22±0.1V */ LVIM = 0b00000000; /* LVI動作禁止 */ /* 電源電圧(VDD)<検出電圧(VLVI)時に割込発生 */ /* 電源電圧(VDD)≧検出電圧