お世話になっております。GAです。
以前、似た内容で一度ご質問させて頂いていたのですが、(community-ja.renesas.com/.../51738このソースコードを実装したファームウェアにて、セルフプログラム中に、とあるコードフラッシュのブロックの先頭アドレスの書き込み直後に、FASTATレジスタに0x10(コマンドロック状態のビットのみ1)とFSTATRレジスタに0x00009000(フラッシュレディフラグとプログラムエラービットのみ1)が発生し、プログラムに失敗します。その際にFPESTATレジスタを読み出すと、0x02(ロックビット以外の原因によるプログラムエラー)が読み出せます。このプログラムエラーが発生するメモリアドレスが、0xfff48000と0xfff50000で確認されており、起動毎や直接関係ない箇所の修正でも発生しており、恐らく一定の位置ではないように思われます。
ここまでの状況を踏まえ、まず単純にイレースが失敗している可能性を考えた為、コードフラッシュ全域のイレース処理を終えた後に、ブランクチェックを行ったのですが、全てのブロックでFBCSTATレジスタが0x0(ブランク)であることが確認されました。また、割り込みで処理が中断されている可能性も考え、プログラムの際ににシリアル通信経由でデータを受け取る必要がある都合上、SCI以外の割り込みを__set_ipl(0xe)で制限し、また、FWEPRORレジスタがウォッチドッグタイマリセット(WDTカウントのリセットと理解しました。)で初期化されてしまう為、FWEPRORレジスタを監視し、許可以外に変化した場合、許可に設定しなおすようにしております。これらを追加した後でも、上記のようにプログラムエラーとなってしまいます。
現状他に思い付く原因が無く、何故失敗するかが分からない状態となっております。お忙しいところ恐縮ですが、お知恵をお借り出来ましたら、幸いです。以下開発環境と処理部、定数部のソースコードの抜粋となります。
[開発環境]ソフト:CS+ for CC V8.11.00 [30 Nov 2023]デバッガー:E2 Lite対象:RX660 (R5F56609EDFP#10)
※下記に記載しているソースコードは 全てがRAM上に展開されて動作していることを確認済みです。
[開発環境]ソフト:CS+ for CC V8.11.00 [30 Nov 2023]デバッガー:E2 Lite対象:RX660 (R5F56609EDFP#10)※下記に記載しているソースコードは 全てがRAM上に展開されて動作していることを確認済みです。(ソースコードにつきまして、どうしてもフォーラム上で、インデントを再現出来なかった為、別途インデント付のtxtファイルを同梱致します。よろしければ、そちらも合わせてご確認ください。)/cfs-file/__key/communityserver-discussions-components-files/69/_BB30EB30D530D730ED30B030E930DF30F330B030BD30FC30B930B330FC30C930E890_.txt
定数部const struct{ UINT bit; UINT* adr;} ERASEBLOCK[] ={ { 0, ( UINT* )0xFFFFE000}, { 1, ( UINT* )0xFFFFC000 }, { 2, ( UINT* )0xFFFFA000 }, { 3, ( UINT* )0xFFFF8000 }, { 4, ( UINT* )0xFFFF6000 }, { 5, ( UINT* )0xFFFF4000 }, { 6, ( UINT* )0xFFFF2000 }, { 7, ( UINT* )0xFFFF0000 }, { 8, ( UINT* )0xFFFE8000 }, { 9, ( UINT* )0xFFFE0000}, { 10, ( UINT* )0xFFFD8000},{ 11, ( UINT* )0xFFFD0000}, { 12, ( UINT* )0xFFFC8000},{ 13, ( UINT* )0xFFFC0000}, { 14, ( UINT* )0xFFFB8000},{ 15, ( UINT* )0xFFFB0000}, { 16, ( UINT* )0xFFFA8000},{ 17, ( UINT* )0xFFFA0000}, { 18, ( UINT* )0xFFF98000},{ 19, ( UINT* )0xFFF90000}, { 20, ( UINT* )0xFFF88000},{ 21, ( UINT* )0xFFF80000}, { 22, ( UINT* )0xFFF78000},{ 23, ( UINT* )0xFFF70000}, { 24, ( UINT* )0xFFF68000},{ 25, ( UINT* )0xFFF60000}, { 26, ( UINT* )0xFFF58000},{ 27, ( UINT* )0xFFF50000}, { 28, ( UINT* )0xFFF48000},{ 29, ( UINT* )0xFFF40000}, { 30, ( UINT* )0xFFF38000},{ 31, ( UINT* )0xFFF30000}, { 32, ( UINT* )0xFFF28000},{ 33, ( UINT* )0xFFF20000}, { 34, ( UINT* )0xFFF18000},{ 35, ( UINT* )0xFFF10000}, { 36, ( UINT* )0xFFF08000},{ 37, ( UINT* )0xFFF00000}};#define BLK_SIZE 128 // 1ブロックサイズ#define DATSIZE BLK_SIZE+6 // 一回のデータ送信サイズ#define STR_BLK 0 // 書込み開始ブロック番号#define END_BLK 37 // 書込み終了ブロック番号#define STR_ADR 0xFFF00000 // 書込み開始アドレス#define END_ADR 0xFFFFFEFF // 書込み終了アドレス処理部//書き込みメインプログラムvoid RomProgSet( void ){ int ret,jk,hige; UCHR buf = 0; ULNG lRcv_Counter = 0L; long checksum; int SequenceNumber = 1; UCHR DumpDates[(BLK_SIZE*5)+1]; UCHR *DumpAddr; UCHR hogechar =0; //デバッグ用レジスタ読み出しバッファ unsigned long hogelong = 0; jk=0; for(jk=0;jk<BLK_SIZE;jk++) { DumpDates[jk]=0x81; } for(jk=0;jk<BLK_SIZE;jk++) { DumpAddr = 0xffff4000 + jk; DumpDates[jk] = *DumpAddr; } do { switch ( SequenceNumber ) { case 1: // 書込み開始 status = OFF; WaitTimer( 20 ); SequenceNumber = 20; break; case 20: // ROMの指定領域消去 hogechar = FLASH.FASTAT.BYTE; hogelong = FLASH.FSTATR.LONG; FLASH.FENTRYR.WORD = 0xAA01;//コードフラッシュ hogechar = FLASH.FASTAT.BYTE; hogelong = FLASH.FSTATR.LONG; do { WaitTimer(1); } while(FLASH.FENTRYR.WORD != 0x0001); do { WaitTimer(1); FLASH.FPROTR.WORD = 0x5501; } while(FLASH.FPROTR.WORD == 0x0000); hogechar = FLASH.FASTAT.BYTE; hogelong = FLASH.FSTATR.LONG; do { WaitTimer(1); }while(FLASH.FSTATR.BIT.FRDY == OFF); FLASH.FWEPROR.BYTE = 0x01; do { FLASH.FWEPROR.BYTE = 0x01; WaitTimer(1); }while( FLASH.FWEPROR.BYTE != 0x01 ); FLASH.FCPSR.WORD = 0x01; hogechar = FLASH.FASTAT.BYTE; hogelong = FLASH.FSTATR.LONG; ret = EraseRom( STR_BLK, END_BLK ); FLASH.FPROTR.WORD = 0x5500; FLASH.FENTRYR.WORD = 0xAA00; for(hige=0;hige<19;hige++) { for(jk=0;jk<=BLK_SIZE*2;jk++) { DumpAddr = STR_ADR + jk+((BLK_SIZE*2+1)*hige); DumpDates[jk] = *DumpAddr; } WaitTimer(1); } hige = 0; jk =0; if ( ret != 0 ) { // 消去エラー status = 1; SequenceNumber = 999; break; } do { WaitTimer(1); }while(FLASH.FSTATR.BIT.FRDY == OFF); SequenceNumber = 30; break; case 30: // ROMの指定領域書込み FLASH.FENTRYR.WORD = 0xAA01; do { WaitTimer(1); } while(FLASH.FENTRYR.WORD == 0x0000); do { WaitTimer(1); FLASH.FPROTR.WORD = 0x5501; } while(FLASH.FPROTR.WORD == 0x0000); do { WaitTimer(1); }while(FLASH.FSTATR.BIT.FRDY == OFF); FLASH.FWEPROR.BYTE = 0x01; hogechar = FLASH.FASTAT.BYTE; hogelong = FLASH.FSTATR.LONG; IEN(SCI1, TXI1) = 1U; ICU.GENBL0.BIT.EN2 = 1U; IEN(SCI1, RXI1) = 1U; ICU.GENBL0.BIT.EN3 = 1U; FLASH.FWEPROR.BYTE = 0x01; ret = ProgRom( STR_ADR, END_ADR ); if ( ret == -1 ) { // 書込みエラー status = 1; SequenceNumber = 999; break; } hogechar = FLASH.FASTAT.BYTE; hogelong = FLASH.FSTATR.LONG; FLASH.FPROTR.WORD = 0x5500; FLASH.FENTRYR.WORD = 0xAA00; do { WaitTimer(1); }while(FLASH.FSTATR.BIT.FRDY == OFF); for(hige=0;hige<19;hige++) { for(jk=0;jk<=BLK_SIZE*2;jk++) { DumpAddr = STR_ADR + jk+((BLK_SIZE*2+1)*hige); DumpDates[jk] = *DumpAddr; } WaitTimer(1); } SequenceNumber = 35; break; case 35: checksum = H8S_CheckSum(); SumSend( checksum ); SequenceNumber = 40; break; case 40: // 終了確認 if ( status == 0 ) // 書込み終了 { lRcv_Counter = 0L; SequenceNumber = 50; } else // 書込みエラー { SequenceNumber = 999; } break; case 50: // チェックサムチェック結果待ち //デバッグ用に一旦コメントアウト buf = 0;// if ( SCI1.SSR.BIT.RDRF ) // 受信データあり// {// buf = SCI1.RDR; // データ読み込み// SCI1.SSR.BIT.RDRF = 0;// }// // チェックサム正常のためリブート// if ( buf == STS_RB )// {// SequenceNumber = 60;// }// // チェックサム異常のためリトライ// if ( buf == STS_WR )// {// SequenceNumber = 1;// } // 旧バージョンにバージョンDOWNのとき強制リブート lRcv_Counter++;// if ( lRcv_Counter >= 800000L ) // タイムアップ(200ms) if ( lRcv_Counter >= 1L ) // タイムアップ(200ms)//デバッグ用に縮小 { SequenceNumber = 70; } break; case 60: //ログデータ送信 SequenceNumber = 70; break; case 70: // 自動リブート for(jk=0;jk<BLK_SIZE*5;jk++) { DumpAddr = STR_ADR + jk; DumpDates[jk] = *DumpAddr; } ICU.NMIER.BIT.WDTEN = 1U; R_BSP_SETPSW_I(); for ( ;; ); // 無限ループによる強制リセット break; case 999: lRcv_Counter = 0L; SequenceNumber = 50; break; default: break; } } while ( SequenceNumber != 0 );}/*----------------------------------------------------------- 機能 :フラッシュROMの指定領域を消去する 引数 :int startBlock: 消去開始アドレス :int endBlock : 消去終了アドレス 戻り値 :正常:(0)/異常:(1)-----------------------------------------------------------*/int EraseRom( int startBlock, int endBlock ){ int block, status; WaitTimer( 20 ); // for ( block = startBlock; block < endBlock; ++block ) for ( block = endBlock; block >= startBlock; --block ) { R_Config_WDT_Restart(); //WDリセット if(FLASH.FENTRYR.WORD == 0x0000){FLASH.FENTRYR.WORD = 0xAA01;} if(FLASH.FPROTR.WORD == 0x0000) { WaitTimer(1); FLASH.FPROTR.WORD = 0x5501; } status = BlockErase( block ); WaitTimer( 5 ); if ( status != 0 ) { break; } status = BlockBlankCheck(block); if ( status != 0 ) { block--; } } WaitTimer( 600 ); return status;}/*----------------------------------------------------------- 機能 :指定ブロックを消去する 引数 :int block: 消去ブロック番号 戻り値 :正常:(0)/異常:(1)-----------------------------------------------------------*/int BlockErase( int block ){ int status, n; bool Endflg = false; UCHR hogechar =0; unsigned long hogelong = 0; unsigned int *hogeintpointer; n=0; FLASH.FSADDR.LONG = ERASEBLOCK[block].adr; hogechar = FLASH.FASTAT.BYTE; hogelong = FLASH.FSTATR.LONG; do { if(FLASH.FSTATR.BIT.FRDY == ON) { if(FLASH.FWEPROR.BYTE != 0x01) { FLASH.FWEPROR.BYTE = 0x01; } FACI = 0x20; FACI = 0xD0; hogechar = FLASH.FASTAT.BYTE; hogelong = FLASH.FSTATR.LONG; if(FACI_command_chk(0xD0)==0) { Endflg = true; } } else{} }while(Endflg == false); Endflg = false; WaitTimer(10); do { if(FLASH.FSTATR.BIT.FRDY == ON) { Endflg = true; if (FLASH.FASTAT.BIT.CMDLK == OFF) { status = 0; } else { status = 1; } } else if(FLASH.FSTATR.BIT.FRDY == OFF) { if(n >= 600) { Endflg = true; status = 1; ForceEndCmd(); } else { n++;} WaitTimer(5000); } }while(Endflg == false); hogechar = FLASH.FASTAT.BYTE; hogelong = FLASH.FSTATR.LONG; return status;}//ブランクチェック関数 // 戻り値 :正常:(0)/異常:(1)int BlockBlankCheck(int block){ int status, n; bool Endflg = false; UCHR hogechar =0; unsigned long hogelong = 0; unsigned int *hogeintpointer; n=0; FLASH.FSADDR.LONG = ERASEBLOCK[block].adr; if(block == 0) { FLASH.FEADDR.LONG = 0xFFFFFFFF; } else { FLASH.FEADDR.LONG = (ERASEBLOCK[block+1].adr)-1; } hogechar = FLASH.FASTAT.BYTE; hogelong = FLASH.FSTATR.LONG; do { if(FLASH.FSTATR.BIT.FRDY == ON) { if(FLASH.FWEPROR.BYTE != 0x01) { FLASH.FWEPROR.BYTE = 0x01; } FACI = 0x71; FACI = 0xD0; hogechar = FLASH.FASTAT.BYTE; hogelong = FLASH.FSTATR.LONG; if(FACI_command_chk(0xD0)==0) { Endflg = true; } } else{} }while(Endflg == false); Endflg = false; WaitTimer(10); do { if(FLASH.FSTATR.BIT.FRDY == ON) { Endflg = true; if (FLASH.FBCSTAT.BYTE == 0x00) { status = 0; } else { status = 1; } } else if(FLASH.FSTATR.BIT.FRDY == OFF) { if(n >= 600) { Endflg = true; status = 1; ForceEndCmd(); } else { n++;} WaitTimer(5000); } }while(Endflg == false); hogechar = FLASH.FASTAT.BYTE; hogelong = FLASH.FSTATR.LONG; return status;}/*----------------------------------------------------------- 機能 :フラッシュROMの指定領域書込み 引数 :ULNG startAdr: 書込み開始アドレス :ULNG endAdr : 書込み終了アドレス 戻り値 :正常:(0)/異常:(-1)-----------------------------------------------------------*/int ProgRom( ULNG startAdr, ULNG endAdr ){ unsigned long* addr; unsigned long count; int i; for ( count = startAdr; count < endAdr; count += 256 ) { addr = count; if( FLASH.FWEPROR.BYTE != 0x01){FLASH.FWEPROR.BYTE = 0x01;} if ( ProgBlock( addr ) != 0 ) { return -1; } i++; } return 0;}/*----------------------------------------------------------- 機能 :1ブロック書込み 引数 :UCHR* Waddr: 戻り値 :正常:(0)/異常:(-1)-----------------------------------------------------------*/int ProgBlock( UCHR* Waddr){ UCHR* dataAddr; UCHR ReWrite[BLK_SIZE + 2], WriteData[BLK_SIZE + 2]; SUCHR program_data[DATSIZE + 2]; int n, prgStatus, i,ErrCount; bool Loopflg,Loopflg2; Loopflg = false; Loopflg2 = false; ErrCount=0; UCHR hogechar =0; unsigned long hogelong = 0; unsigned int hogeint = 0; unsigned long addr_memo = 0; //本来のデータ受け取り関数 デバッグ中により、コメントアウト // 新しいプログラムデータを取得(134byte) // if ( getProgDat( program_data ) != 0 ) // { // return ( 1 ); // } //デバッグ用のデータ受け取り関数 ↑と同じデータの一部をヘッターファイルに入れて代用 if ( getProgDat_Dummy( program_data ) != 0 ) { return ( 1 ); } // 配列の3~130番目が実プログラムデータ */ dataAddr = ( UCHR* )( &program_data[3] ); for ( i = 0; i <= BLK_SIZE+1; ++i, ++dataAddr ) { WriteData[i] = ReWrite[i] = *dataAddr; } FLASH.FWEPROR.BYTE = 0x01; FLASH.FSADDR.BIT.FSADDR = Waddr; addr_memo= FLASH.FSADDR.BIT.FSADDR; do { WaitTimer(1); } while(FLASH.FSTATR.BIT.DBFULL == ON); FACI = 0xE8; FACI = 0x80; hogechar = FLASH.FASTAT.BYTE; hogelong = FLASH.FSTATR.LONG; FACI_S = (unsigned short)WriteData[4] | ( ((unsigned short)WriteData[3]) <<8 ); hogechar = FLASH.FASTAT.BYTE; hogelong = FLASH.FSTATR.LONG; n=1; do { FLASH.FWEPROR.BYTE = 0x01; ); FACI_S = (unsigned short)WriteData[(2*n)+4] | ( ((unsigned short)WriteData[(2*n)+3]) <<8 ); hogechar = FLASH.FASTAT.BYTE; hogelong = FLASH.FSTATR.LONG; if(n==127) { hogechar = FLASH.FASTAT.BYTE; hogelong = FLASH.FSTATR.LONG; while(FLASH.FSTATR.BIT.DBFULL == ON) { hogechar = FLASH.FASTAT.BYTE; hogelong = FLASH.FSTATR.LONG; }; FLASH.FWEPROR.BYTE = 0x01; FACI = 0xD0; hogechar = FLASH.FASTAT.BYTE; hogelong = FLASH.FSTATR.LONG; while(FLASH.FSTATR.BIT.FRDY != 1) { __nop(); } FACI_command_chk(0xD0); Loopflg = true; hogechar = FLASH.FASTAT.BYTE; hogelong = FLASH.FSTATR.LONG; if(hogechar == 0x10 || hogelong != 0x00008000) { hogeint = FLASH.FPESTAT.WORD; } } else if (FLASH.FSTATR.BIT.DBFULL == ON) { while(Loopflg2 == false) { ErrCount++; WaitTimer(4); if(FLASH.FSTATR.BIT.DBFULL == OFF) { ErrCount = 0; Loopflg2 =true; hogechar = FLASH.FASTAT.BYTE; hogelong = FLASH.FSTATR.LONG; } else if (ErrCount == 4) { ForceEndCmd(); Loopflg = true; Loopflg2 = true; prgStatus = -1; hogechar = FLASH.FASTAT.BYTE; hogelong = FLASH.FSTATR.LONG; } } } n++; R_Config_WDT_Restart(); }while(Loopflg == false); Loopflg = false; while(Loopflg == false) { if(FLASH.FSTATR.BIT.FRDY == 1) { Loopflg = true; } else { hogechar = FLASH.FASTAT.BYTE; hogelong = FLASH.FSTATR.LONG; ErrCount++; WaitTimer(80); if(ErrCount == 4) { Loopflg = true; ForceEndCmd(); prgStatus = -1; } } } n=0; prgStatus = 0; hogechar = FLASH.FASTAT.BYTE; hogelong = FLASH.FSTATR.LONG; return prgStatus;}/*----------------------------------------------------------- 機能 :ウェイトタイマー 引数 :UINT time: ループ回数 戻り値 :無し-----------------------------------------------------------*/void WaitTimer( UINT time ){ UINT i; int hoge = 0; for ( i = 0; i < time; i++ ) { R_Config_WDT_Restart(); //WDリセット // nop(); hoge = (i*i)/(i-1); }}/************************************************************************************************************************ Function Name: R_Config_WDT_Restart* Description : This function restarts WDT module* Arguments : None* Return Value : None***********************************************************************************************************************/void R_Config_WDT_Restart(void){ /* Refreshed by writing 00h and then writing FFh */ WDT.WDTRR = 0x00U; WDT.WDTRR = 0xFFU;}
cacao99様お世話になっております。上記の件、ご連絡ありがとうございます。内容確認させていただきました。フォーラムのメンテナンスが始まってしまう為、取り急ぎ、お礼だけ先にご連絡させていただきます。ありがとうございます。頂いた内容で少し自分なりに考えてみます。以上です。宜しくお願い致します。