質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

ただいまの
回答率

88.77%

C#/Windows フォームアプリケーションでのSystem.AccessViolationExceptionについて

解決済

回答 1

投稿

  • 評価
  • クリップ 1
  • VIEW 2,120

Kashi_wamochi

score 14

前提・実現したいこと

現在C#で計測制御のプログラムを作成しています。
使用するのはContecのPCIe接続のアナログ入力ボードで、現物が手元にないため、
API-AIO(WDM)からDemo Deviceという仮想デバイスを用いて開発しています。

環境はVisual Studio 2015/Windows フォームアプリケーション/.Net Framework 4.6.1です。

まず取っ掛かりとして、100ミリ秒間隔で16個のテキストボックスにアナログ入力の値を表示しようとしていましたが、
20秒程度プログラムが動いたのち、以下のエラーメッセージが発生しました。

発生している問題・エラーメッセージ

System.AccessViolationException
HResult=0x80004003
Message=保護されているメモリに読み取りまたは書き込み操作を行おうとしました。他のメモリが壊れていることが考えられます。
Source=System.Windows.Forms
スタック トレース:
場所 System.Windows.Forms.PropertyStore.LocateObjectEntry(Int16 entryKey, Int32& index)
場所 System.Windows.Forms.PropertyStore.GetObject(Int32 key, Boolean& found)
場所 System.Windows.Forms.Form.get_IsMdiChild()
場所 System.Windows.Forms.Form.CanRecreateHandle()
場所 System.Windows.Forms.Form.set_Active(Boolean value)
場所 System.Windows.Forms.Form.WmActivate(Message& m)
場所 System.Windows.Forms.Form.WndProc(Message& m)
場所 System.Windows.Forms.Control.ControlNativeWindow.OnMessage(Message& m)
場所 System.Windows.Forms.Control.ControlNativeWindow.WndProc(Message& m)
場所 System.Windows.Forms.NativeWindow.DebuggableCallback(IntPtr hWnd, Int32 msg, IntPtr wparam, IntPtr lparam)

該当のソースコード

using System;
using System.Collections.Generic;
using System.Linq;
using System.Threading.Tasks;
using System.Windows.Forms;

namespace DigitShowBasic
{
    static class Program
    {
        /// <summary>
        /// アプリケーションのメイン エントリ ポイントです。
        /// </summary>
        [STAThread]
        static void Main()
        {
            Application.EnableVisualStyles();
            Application.SetCompatibleTextRenderingDefault(false);
            Application.Run(new MainForm()); // ここでエラーメッセージが発生
        }
    }
}
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Windows.Forms;
using CaioCs;

namespace DigitShowBasic
{
    public partial class MainForm : Form
    {
        Caio aio = new Caio();

        int Ret;
        short m_Id;
        DateTime StartTime;
        static short ADNUM = 16;
        int ChartPointNum = 1000;
        int AiSamplingTimes = ADNUM;
        float[] AiData = new float[ADNUM];

        public MainForm()
        {
            InitializeComponent();

            AdInitialize();
        }

        private void AdInitialize()
        {
            Ret = aio.Init("Aio000", out m_Id);
            ErrorJudge(Ret);
            Ret = aio.SetAiInputMethod(m_Id, 0);
            ErrorJudge(Ret);
            Ret = aio.SetAiChannels(m_Id, ADNUM);
            ErrorJudge(Ret);
            Ret = aio.SetAiRangeAll(m_Id, 0);
            ErrorJudge(Ret);
            Ret = aio.SetAiTransferMode(m_Id, 0);
            ErrorJudge(Ret);
            Ret = aio.SetAiMemoryType(m_Id, 0);
            ErrorJudge(Ret);
            Ret = aio.SetAiClockType(m_Id, 0);
            ErrorJudge(Ret);
            Ret = aio.SetAiSamplingClock(m_Id, 100000);
            ErrorJudge(Ret);
            Ret = aio.SetAiStartTrigger(m_Id, 0);
            ErrorJudge(Ret);
            Ret = aio.SetAiStopTrigger(m_Id, 4);
            ErrorJudge(Ret);
            Ret = aio.ResetAiMemory(m_Id);
            ErrorJudge(Ret);
            int AiSamplingCount;
            Ret = aio.GetAiSamplingCount(m_Id, out AiSamplingCount);
            ErrorJudge(Ret);
            Ret = aio.StartAi(m_Id);
            ErrorJudge(Ret);
            StartTime = DateTime.Now;

        }

        private void MainForm_Closed(object sender, FormClosedEventArgs e)
        {
            Ret = aio.StopAi(m_Id);
            ErrorJudge(Ret);
            Ret = aio.Exit(m_Id);
            ErrorJudge(Ret);
        }

        private void timer1_Tick(object sender, EventArgs e)
        {
            Ret = aio.GetAiSamplingDataEx(m_Id, ref AiSamplingTimes, ref AiData);
            CH01_textbox.Text = AiData[0].ToString("F6");
            CH02_textbox.Text = AiData[1].ToString("F6");
            CH03_textbox.Text = AiData[2].ToString("F6");
            CH04_textbox.Text = AiData[3].ToString("F6");
            CH05_textbox.Text = AiData[4].ToString("F6");
            CH06_textbox.Text = AiData[5].ToString("F6");
            CH07_textbox.Text = AiData[6].ToString("F6");
            CH08_textbox.Text = AiData[7].ToString("F6");
            CH09_textbox.Text = AiData[8].ToString("F6");
            CH10_textbox.Text = AiData[9].ToString("F6");
            CH11_textbox.Text = AiData[10].ToString("F6");
            CH12_textbox.Text = AiData[11].ToString("F6");
            CH13_textbox.Text = AiData[12].ToString("F6");
            CH14_textbox.Text = AiData[13].ToString("F6");
            CH15_textbox.Text = AiData[14].ToString("F6");
            CH16_textbox.Text = AiData[15].ToString("F6");
        }

        private void ErrorJudge(int Ret)
        {
            if (Ret != 0)
            {
                string ErrorString;
                aio.GetErrorString(Ret, out ErrorString);
                MessageBox.Show("aio.Init = " + Ret.ToString() + " : " + ErrorString);
                return;
            }
            return;
        }
    }

}

試したこと

Access violationに惑わされて4時間奪われた話
こちらを参考に、「「プロジェクトのプロパティ>デバッグ>ネイティブ コードのデバッグを有効にする(T)」にチェック」をやってみたのですが、症状は変わらずでした。

補足情報(FW/ツールのバージョンなど)

teratail初質問になります。よろしくお願いいたします。

  • 気になる質問をクリップする

    クリップした質問は、後からいつでもマイページで確認できます。

    またクリップした質問に回答があった際、通知やメールを受け取ることができます。

    クリップを取り消します

  • 良い質問の評価を上げる

    以下のような質問は評価を上げましょう

    • 質問内容が明確
    • 自分も答えを知りたい
    • 質問者以外のユーザにも役立つ

    評価が高い質問は、TOPページの「注目」タブのフィードに表示されやすくなります。

    質問の評価を上げたことを取り消します

  • 評価を下げられる数の上限に達しました

    評価を下げることができません

    • 1日5回まで評価を下げられます
    • 1日に1ユーザに対して2回まで評価を下げられます

    質問の評価を下げる

    teratailでは下記のような質問を「具体的に困っていることがない質問」、「サイトポリシーに違反する質問」と定義し、推奨していません。

    • プログラミングに関係のない質問
    • やってほしいことだけを記載した丸投げの質問
    • 問題・課題が含まれていない質問
    • 意図的に内容が抹消された質問
    • 過去に投稿した質問と同じ内容の質問
    • 広告と受け取られるような投稿

    評価が下がると、TOPページの「アクティブ」「注目」タブのフィードに表示されにくくなります。

    質問の評価を下げたことを取り消します

    この機能は開放されていません

    評価を下げる条件を満たしてません

    評価を下げる理由を選択してください

    詳細な説明はこちら

    上記に当てはまらず、質問内容が明確になっていない質問には「情報の追加・修正依頼」機能からコメントをしてください。

    質問の評価を下げる機能の利用条件

    この機能を利用するためには、以下の事項を行う必要があります。

質問への追記・修正、ベストアンサー選択の依頼

  • Kashi_wamochi

    2019/04/11 16:11

    >hihijiji様
    ありがとうございます。とりあえず使用したモジュールのドキュメントを参照しながら、引数があっているかどうかを確認してみます。

    キャンセル

  • Kashi_wamochi

    2019/04/11 16:40

    >dodox様
    1. Windows10 Pro 1809です
    2. 結果的にこれが正解でした。この関数の呼び出し時にエラーが出ていました。エラーの内容はメモリのオーバーフローです。GetAiSamplingDataExにおいて、読み込むデータの数を表す第2引数が、第3引数の配列の大きさよりも大きい値を入力していました。
    今後読み出し間隔をさらに狭めることも考えていたので、少しでもTick内の処理を軽くしようとして、エラーチェックを外していたのですが、これが盲点でした。
    的確なコメント、ありがとうございます。

    キャンセル

  • hihijiji

    2019/04/11 16:43

    ここに書かずに、解答欄に書いて解決済みにしてくださるようお願いします。

    キャンセル

回答 1

check解決した方法

0

Ret = aio.GetAiSamplingDataEx(m_Id, ref AiSamplingTimes, ref AiData);


GetAiSamplingDataExにおいて、読み込むデータの数を表すAiSamplingTimesが、AiDataの配列の大きさよりも大きい値を入力していました。コメントいただいたdodox様、ありがとうございました。

投稿

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

15分調べてもわからないことは、teratailで質問しよう!

  • ただいまの回答率 88.77%
  • 質問をまとめることで、思考を整理して素早く解決
  • テンプレート機能で、簡単に質問をまとめられる

関連した質問

同じタグがついた質問を見る