ReadFileでエラーが出ます

このQ&Aのポイント
  • シリアルポートで非同期でバイナリを受信するプログラムを作成しています。
  • 受信部分は以下のソースです。これで実行すると「データ受信エラー」のダイアログが出ます。
  • GetLastErrorでは998(メモリロケーションへのアクセスが無効です。)が得られました。
回答を見る
  • ベストアンサー

ReadFileでエラーが出ます。

シリアルポートで非同期でバイナリを受信するプログラムを作成しています。 受信部分は以下のソースです。これで実行すると「データ受信エラー」のダイアログが出ます。 GetLastErrorでは998(メモリ ロケーションへのアクセスが無効です。)が得られました。 ネットで調べましたが何が原因かも分かりません・・。 どうかアドバイスお願いします。 HANDLE hEvent; hEvent = CreateEvent( NULL, FALSE, FALSE, NULL ); OVERLAPPED old; ZeroMemory( &old, sizeof(old) ); old.Offset = 0; old.OffsetHigh = 0; old.hEvent = hEvent; DWORD dwCount; char* rdBuf; DWORD dwRead; if(!ReadFile(hCom,rdBuf,dwCount,&dwRead,&old)){ if(ERROR_IO_PENDING == GetLastError()){ GetOverlappedResult(hCom,&old,&dwRead,TRUE); } else MessageBox("データ受信エラー","試作1",MB_ICONSTOP); }

質問者が選んだベストアンサー

  • ベストアンサー
  • tokichim
  • ベストアンサー率42% (88/205)
回答No.3

> 以下のように書いて実行するとエラーが1つでるのですが、何か間違えていますでしょうか? 失礼。rdBuf変数が配列とポインタで名前がかぶってましたね。 どっちかの名前を変えてください。 #define RDBUF_SIZ 1 char rdDataBuf[RDBUF_SIZ]; char *rdBuf; rdBuf = &rdDataBuf[0]; dwCount = RDBUF_SIZ;

meeyooyoo
質問者

お礼

有難うございました! うまくできました。 tokichimさん、本当に有難うございました。

その他の回答 (2)

  • tokichim
  • ベストアンサー率42% (88/205)
回答No.2

> rdBufとdwCountにはどんな値を入れたらいいのでしょうか? 正直いうと、このくらいはReadFile()のマニュアルを見てほしいのですが。 一応答えると、データを格納するバッファへのポインタと、読み込もうとするサイズです。 1バイトだけ読み込みたいのなら、 #define RDBUF_SIZ 1 char rdbuf[RDBUF_SIZ]; rdbuf = &rdbuf[0]; dwCount = RDBUF_SIZ; みたいな感じ。

meeyooyoo
質問者

補足

tokichimさん、有難うございます。 以下のように書いて実行するとエラーが1つでるのですが、何か間違えていますでしょうか? #define RDBUF_SIZ 1 char rdBuf[RDBUF_SIZ]; rdBuf = &rdBuf[0]; DWORD dwCount; dwCount = RDBUF_SIZ; (3行目)error C2440: '=' : 'char *' から 'char [1]' に変換することはできません。

  • tokichim
  • ベストアンサー率42% (88/205)
回答No.1

rdBufとdwCountに値を入れてないようですが。

meeyooyoo
質問者

補足

有難う御座います! rdBufとdwCountにはどんな値を入れたらいいのでしょうか?

関連するQ&A

  • バイナリを16進数で表示したい

    VC++6.0,MFC,ダイアログベースでバイナリデータ受信とファイル書き込みのプログラムを作っています。 COMポートからバイナリを10バイト受信し、受信した内容を1.txtに書き込み、1.txtを開いて、受信した内容を確認したいです。 以下のソースで実行すると、1.txtにはFフフフフフフフフフと書かれていました。これを16進数で表示させたり、10進数で表示させるにはどうしたらいいでしょうか? /////////データ受信////////////// unsigned char rdBuf[10]; unsigned char* prdBuf; DWORD dwCount; DWORD dwRead; prdBuf = &rdBuf[0]; dwCount = 10; if(ReadFile(hCom,prdBuf,dwCount,&dwRead,&old) == 0){ if(ERROR_IO_PENDING == GetLastError()){ if(WaitForSingleObject(hEvent,INFINITE) == WAIT_OBJECT_0){ if(GetOverlappedResult(hCom,&old,&dwCount,TRUE)){ //"データ受信完了" }}} else //エラー } /////////ファイルに書き込む////////////// HANDLE hFile; hFile = CreateFile( ".\\ScanData\\1.txt", GENERIC_WRITE, 0, NULL, OPEN_ALWAYS, FILE_FLAG_OVERLAPPED, NULL ); if(WriteFile( hFile, rdBuf,10, &wbyte, &old ) == 0){ if(ERROR_IO_PENDING == GetLastError()){ if(WaitForSingleObject(hEvent,INFINITE)==WAIT_OBJECT_0){ if(GetOverlappedResult(hCom,&old,&wbyte,TRUE)){ //書き込み完了 }}} else //書き込み失敗 }

  • 可変長バイナリを読み込みたい

    <プログラム環境> Windows XP,VC++6.0,MFC AppWizard(exe),ダイアログベース <目的> COMポートから可変長のバイナリデータを受信する <質問> 受信するバイナリの長さは受信した内容で分かるようになっています。 例えば、 先頭が"0x0b"であれば11個のバイナリが続いて送られてくるという感じです。 この例ように、送られてくる内容("0x0b")を確認して、それに応じて 11個データを読み込む、というプログラムにするには、どのようにしたら良いでしょうか? 今は以下のソースで、バイナリの長さを指定して受信しています。 //////////オーバーラップ構造体の初期化////////// OVERLAPPED old; ZeroMemory( &old, sizeof(old) ); old.Offset = 0; old.OffsetHigh = 0; old.hEvent = hEvent; //////////データ受信//////////////////////////// unsigned char rdBuf[10]; unsigned char* prdBuf; DWORD dwCount; DWORD dwRead; prdBuf = &rdBuf[0]; dwCount = 10; if(ReadFile(hCom,prdBuf,dwCount,&dwRead,&old) == 0){ if(ERROR_IO_PENDING == GetLastError()){ if(WaitForSingleObject(hEvent,INFINITE) == WAIT_OBJECT_0){ if(GetOverlappedResult(hCom,&old,&dwCount,TRUE)){ //"データ受信完了" }}} else //エラー } 宜しければ、ご指摘の程よろしくお願いします。

  • バイナリをテキストファイルに書き込みたい

    <プログラム環境> Windows XP,VC++6.0,MFC AppWizard(exe),ダイアログベース <目的> COMポートから受信した可変長バイナリデータを1.txtに書き込む。 ※バイナリエディタを使用せずに16進数で読めるように書く <質問> 受信したバイナリデータは「02 80 9a 00 b1 56」なのですが、これをテキストファイルに16進数で書き込むことができません。 どのようにしたら良いでしょうか? 今は以下のソースです。 コメントアウト部分はMessageBox等で表示する処理をしています。 先頭1バイト受信→"06"を受信したのでさらに6バイト受信→6バイト書き込む。という順序です。 以下で実行するとファイルには「 00000」と書き込まれました。何なのか全く分かりません・・。 //////////先頭の1バイトだけ受信する///////////////////////////////////////// DWORD bLError; unsigned char ucLen; DWORD dwCount = 1; DWORD dwRead; if(ReadFile(hCom,&ucLen,dwCount,&dwRead,&old) == 0){ if(ERROR_IO_PENDING == GetLastError()){ bLError = WaitForSingleObject(hEvent,100); if(bLError == WAIT_OBJECT_0){ if(GetOverlappedResult(hCom,&old,&dwCount,TRUE));//データ受信完了 } else if(bLError == WAIT_FAILED);//受信スリープエラー else if(bLError == WAIT_TIMEOUT);//受信タイムアウト } else ;//受信エラー } //////////先頭データの情報から可変長で読み込む////////////////////////////// unsigned char rdBuf[256]; unsigned char* prdBuf; prdBuf = &rdBuf[0]; dwCount = ucLen; if(ReadFile(hCom,prdBuf,dwCount,&dwRead,&old) == 0){ if(ERROR_IO_PENDING == GetLastError()){ bLError = WaitForSingleObject(hEvent,100); if(bLError == WAIT_OBJECT_0){ if(GetOverlappedResult(hCom,&old,&dwCount,TRUE))));//データ受信完了 } else if(bLError == WAIT_FAILED);//受信スリープエラー else if(bLError == WAIT_TIMEOUT);//受信タイムアウト } else ;//受信エラー } //////////データ書き込み///////////////////////////////////////////////////// HANDLE hFile; hFile = CreateFile( ".\\ScanData\\1.txt", GENERIC_WRITE, 0, NULL, CREATE_ALWAYS, FILE_FLAG_OVERLAPPED,NULL ); char Buf[256]; char wBuf[256] = ""; wsprintf(Buf,"%#02x",rdBuf); for(BYTE i=1;i<dwRead;i++){ wsprintf(Buf,"%#02x ",rdBuf[i]); wBuf[i] = Buf[0]; } if(WriteFile( hFile, wBuf,dwRead, &wbyte, &old ) == 0){ if(ERROR_IO_PENDING == GetLastError()){ if(WaitForSingleObject(hEvent,1000)==WAIT_OBJECT_0){ if(GetOverlappedResult(hCom,&old,&wbyte,TRUE)){//データ書き込み完了 }}} else ;//データ書き込みエラー } //////////受信したデータ10バイト分をMessageBoxで表示////////////////////// wsprintf(Buf,"文字 = %02x %02x %02x %02x %02x %02x %02x %02x %02x %02x",rdBuf[0],rdBuf[1],rdBuf[2],rdBuf[3],rdBuf[4],rdBuf[5],rdBuf[6],rdBuf[7],rdBuf[8],rdBuf[9]); MessageBox(Buf); 宜しければ、ご指摘の程よろしくお願いします。

  • 16進数の表示ができません。

    シリアルポートで受信したバイナリデータを表示させると、 正しい値が出ないのですが、どうしてでしょうか? 受信するデータは16進数で 06 02 81 07 00 bb 01 b4 00 32 なのですが、以下のソースで実行すると、 6 2 ffffff80 7 0 ffffffbb 1 ffffffb4 0 32 と表示されました。 また、「データ受信完了」というメッセージが出ないのですが、 出なくてもデータ受信は出来ているのでしょうか? char szBuf[256]; char rdBuf[10]; char* prdBuf; DWORD dwCount; DWORD dwRead; dwCount = 10; prdBuf = &rdBuf[0]; if(ReadFile(hCom,prdBuf,dwCount,&dwRead,&old) == 0){ if(ERROR_IO_PENDING == GetLastError()){ if(WaitForSingleObject(hEvent,INFINITE); == WAIT_OBJECT_0){ if(GetOverlappedResult(hCom,&old,&dwCount,TRUE)) MessageBox("データ受信完了"); } } else MessageBox("データ受信エラー","試作1",MB_ICONSTOP); } wsprintf(szBuf,"文字 = %x %x %x %x %x %x %x %x %x %x", rdBuf[0],rdBuf[1],rdBuf[2],rdBuf[3],rdBuf[4],rdBuf[5],rdBuf[6],rdBuf[7],rdBuf[8],rdBuf[9]); MessageBox(szBuf); よろしくお願いします。

  • WriteFileで送信できたかの確認方法は?

    シリアルポートで非同期でバイナリを送信するプログラムを作成しています。 以下のソースでバイナリを送信したつもりなのですが、本当に送信できているのか分りません。 送信が完了したのか確認するにはどうしたいいでしょうか? よろしくお願いします。 COMのオープン、DCBの設定は恐らく問題ないので、 WriteFileの周辺を抜粋します。 OVERLAPPED old; ZeroMemory( &old, sizeof(old) ); old.Offset = 0; old.OffsetHigh = 0; old.hEvent = NULL; char wbuf[7] = {0x02,0x0C,0x01,0x00,0x31,0x15,0x1A}; DWORD wbyte; if(!WriteFile(hCom,wbuf,(DWORD)sizeof(wbuf),&wbyte,&old)){ if(ERROR_IO_PENDING == GetLastError()){ GetOverlappedResult(hCom,&old,&wbyte,TRUE); } else MessageBox("エラー"); }

  • ReadFileについて

    はじめまして。 コマンドラインから実行可能なRS232C通信プログラムを Cで作成しています。 ハイパーターミナルからCOM2に文字を出力して ReadFileで取得した文字列を画面にはき出すというだけの 「Hello,World!」のような簡単なプログラムなのですが ReadFileの動作結果に思うようなものが得られず困っています。 お知恵をお貸しください。 読込結果がおかしいのです。 リターンキー以外は何を入力しても「7F」が返ってきてしまいます (リターンキー入力の場合も正常値ではありません)。 また、入力も無視されたり、認識されたりで動作も不安定です。 ReadFile自体のエラーは発生しません。 よろしくお願いいたします。 Borland C++ Compiler 5.5 Windows2000(実際に動作させたいのはWin95上です) 以下、コードです。 DWORD dwErrors; // エラー情報 COMSTAT ComStat; // デバイスの状態 DWORD dwCount; //受信データのバイト数 DWORD dwRead; // ポートから読み出したバイト数 unsigned char pszBuf[30];// 受信バッファ while(1){  ClearCommError(hComm, &dwErrors, &ComStat);  dwCount = ComStat.cbInQue;  if (0 >= dwCount) continue;  memset(pszBuf,'\0',sizeof(pszBuf));  if (ReadFile(hComm, pszBuf, dwCount, &dwRead, NULL)){   if (dwRead > 0) {    if (pszBuf[0] == '\r') break;    printf("%%s : %s\n",pszBuf);   }else{    break;   }  } }

  • 非同期のプロセス間通信(パイプ)で全データ受信する

    こんにちは。 非同期のプロセス間通信(パイプ)で詰まっています。 環境はWindowsで処理系はC++(Win32)です。 スレッドを用意して、 hPipe = ::CreateFile( L"\\\\.\\pipe\\pipename", GENERIC_WRITE | GENERIC_READ, FILE_SHARE_READ | FILE_SHARE_WRITE, NULL, OPEN_EXISTING, FILE_FLAG_OVERLAPPED, NULL ) ; で作成し、OVERLAPPED構造体を // ZeroMemory( &so, sizeof( OVERLAPPED )) ; sOverlapped.hEvent = ::CreateEvent( NULL, FALSE, FALSE, NULL ) ; として、 const size_t bufsize = 16 ; BYTE     buffer[ uBufSize ] ; DWORD    dwNumberOfBytesRead ; bRet = ::ReadFile( hPipe, buffer, bufsize, &dwNumberOfBytesRead, &so ) ; で読み込みをしています。ReadFile関数はすぐ戻りbRetはFALSEで返ってくるため、GetLastErrorを調べて、 switch( ::GetLastError()) {   case ERROR_IO_PENDING :     bRet = ::GetOverlappedResult( hPipe, &so, &dwNumberOfBytesRead, FALSE ) ;     break ; } としてWaitForMultipleObjectsでsOverlapped.hEventがシグナル状態になったら取得したデータの処理をしています。 シグナル状態になるならないに関わらず、上記のReadFileとGetOverlappedResultはループでぐるぐる回しています。 上記の状態で短いデータならよかったのですが、上記のReadFileで読み込み最大サイズの16バイトを超えてしまとうと、 残りの部分のみしか取得できませんでした。 0123456789ABCDEFabcdefg というデータを受信しようとしたとき、後半のabcdefgだけしか取得できませんでした。 すべてのデータを正しく取得するにはどのようにしたらよいのでしょうか?

  • バイナリ送信で出るエラーコードの意味が分かりません

    シリアルポートでバイナリを非同期で送信するプログラムを作成しています。 以下のソースで実行すると、「重複したI/O処理を実行しています」というエラーがでました。 何が間違っているのか分りません・・。 どうかご指摘お願いします。 void CMyDlg::OnReset() { //COMポートオープン HANDLE hCom; hCom = CreateFile( "COM4", GENERIC_READ | GENERIC_WRITE, 0, NULL, OPEN_EXISTING, FILE_FLAG_OVERLAPPED, NULL ); //通信設定 DCB dcb; GetCommState(hCom,&dcb); dcb.BaudRate = 9600; dcb.ByteSize = 8; dcb.Parity = NOPARITY; dcb.StopBits = ONESTOPBIT; SetCommState(hCom,&dcb); OVERLAPPED old; // オーバーラップ構造体の初期化 ZeroMemory( &old, sizeof(old) ); old.Offset = 0; old.OffsetHigh = 0; old.hEvent = NULL; //データ送信 char wbuf[7] = {0x02,0x00,0x01,0x00,0x31,0x15,0x12}; DWORD wbyte; WriteFile(hCom,wbuf,sizeof(wbuf),&wbyte,&old); //エラーコード取得&文字列に変換&表示 LPVOID lpMessageBuffer; FormatMessage( FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM, NULL, GetLastError(),//エラーコード取得 MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), (LPTSTR) &lpMessageBuffer, 0, NULL ); MessageBox((LPCTSTR)lpMessageBuffer); //メッセージボックス LocalFree( lpMessageBuffer ); //COMポートクローズ CloseHandle(hCom); }

  • WriteFileのエラー995は何が原因?

    <プログラム環境> Windows XP VC++6.0 MFC AppWizard(exe) ダイアログベース <症状> 以下のコード部分は、これまでエラーは無かったのですが、 他の部分をいろいろといじった結果以下のコードでエラーが出るように なりました。 HANDLE hCom;//COMポートのハンドル OVERLAPPED old;//構造体 unsigned char ucBuf[];//送信する内容 DWORD dwByte;//送信するバイト数 //上記変数は宣言後、正しい値を入れています。 BOOL blRet;//実行結果 if(WriteFile(hCom,ucBuf,dwByte,NULL,&old) == 0){ blRet = GetLastError(); } を実行後、blRetの結果を確認すると995で、 「スレッドが終了したか、またはアプリケーション要求によって、I/O 処理が中止されました。」 が出ました。 <質問> このエラーは何が原因でしょうか? 宜しければご指摘の程よろしくお願いします。

  • ReadFileの読み込みエラーについて

     こんばんは Win32 APIでプログラミングをやっているのですが、そのAPIの中でReadFileというものがありますよね? この関数でエラー値が出力され、第二引数としてBITMAPINFO構造体に格納するはずの値がそのままです。 以下に一部ソースを載せます BOOL Cell_Init( HINSTANCE hInstance ) { BITMAPINFO bmi2; HANDLE hFile; DWORD dwBytes; // 作成するビットマップの情報を設定する BITMAPFILEHEADER bmpFileHeader; BITMAPINFO bmi; // ビットマップのハンドルと、作成したバッファの先頭アドレスを取得 void* pImage; HBITMAP hBitmap; // デスクトップのDCを得る HDC tmpDC = GetDC(GetDesktopWindow() ); // HBITMAPにHDCを結びつける g_hdc_cell = CreateCompatibleDC( tmpDC ); DWORD er; hFile = CreateFile("cointos.bmp", GENERIC_READ, 0, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL); er = GetLastError(); if(hFile == INVALID_HANDLE_VALUE) return 0; ReadFile(hFile, &bmpFileHeader, sizeof(BITMAPFILEHEADER), &dwBytes, NULL); er = GetLastError(); if(bmpFileHeader.bfType != 0x4D42) { MessageBox(NULL, TEXT("This is not a bitmap file"), NULL, MB_OK); return FALSE; }       //ここでbmi2に格納される値が格納されていない(そのためでかい値のまま) そのためCreateDIBSectionではエラーを起こす ReadFile(hFile, &bmi2, bmpFileHeader.bfOffBits - dwBytes, &dwBytes, NULL); er = GetLastError(); // ここの部分でエラー値998が出ます hBitmap = CreateDIBSection(tmpDC, &bmi2, DIB_RGB_COLORS, (LPVOID*)&pImage, NULL, 0); ReadFile(hFile,pImage, bmpFileHeader.bfSize - bmpFileHeader.bfOffBits, &dwBytes, NULL); SelectObject( g_hdc_cell, hBitmap ); CloseHandle(hFile); ReleaseDC(GetDesktopWindow(), tmpDC); return TRUE; } なぜこうなるんでしょうか? bmi2はBITMAPINFO構造体でローカルで宣言しています。これがポインタだと大丈夫なんですが なぜ変数ではダメなのか分かりません。 あらかじめ確保されているスタック領域の容量限界でも ないと思うのですが。 分かる方がいらっしゃったらよろしくお願いします。 開発環境はVS.2005 です。

専門家に質問してみよう