私は関数がこのように見えるネイティブDLLに関数を持っています
int GetActiveNames(char* names[]);
文字列/文字の配列を返す上記の関数に対して C# で PInvoke ラッパー メソッドを作成するにはどうすればよいですか?
もっと分かりやすく..
P/Invoke の方法を試してみたところ、次のエラーAttempted to read or write protected memory が表示されました。これは多くの場合、他のメモリが破損していることを示しています。.
簡単にするために、実際の C 関数のシグネチャを取り除いただけです。ただし、完全な C 関数の署名と関数のパラメーターの説明は次のとおりです。
int OpalGetActiveProjects(unsigned short allocatedProjects,
unsigned short *numProjects,
unsigned short allocatedModels,
unsigned short *numModels,
unsigned short allocatedProjectsPathLen,
unsigned short *maxProjectPathLen,
unsigned short allocatedModelPathLen,
unsigned short *maxModelPathLen,
unsigned short allocatedMachineNameLen,
unsigned short *maxMachineNameLen,
unsigned short allocatedIpLen,
unsigned short *maxIpLen,
char *projectPaths[],
unsigned short instanceIDs[],
char *machineNames[],
char *controllerIPs[],
char *modelPaths[],
unsigned short numModelsPerProject[]
unsigned short modelIDs[],
unsigned short modelStates[]);
この関数は、アクティブなプロジェクトのリストを返します。この関数は、プロジェクトごとに、構成ファイルのフル パス、このプロジェクトを開いたマシンの名前と IP アドレス、現在のプロジェクト セッションのインスタンス ID、およびこのプロジェクトに属するモデルのリストを返します。モデルごとに、パス、ID、および状態が返されます。割り当てられたストレージがいずれかの引数に対して小さすぎる場合、リストは返されませんが、numProjects、numModels、maxProjectPathLen、maxModelPathLen、maxMachineNameLen、および maxIpLen の値が設定されます。戻り値は E2BIG です。アプリケーションはこの事実を利用して、最初の呼び出しで長さ 0 を指定し、必要なストレージを割り当て、2 番目の呼び出しを発行して情報を取得できます。
入力パラメータ
allocatedProjects : 呼び出し元によってパス、マシン名、IP アドレス、およびインスタンス ID 用にストレージが割り当てられたプロジェクトの数。 allocatedModels : 名前、パス、およびインスタンス ID 用に呼び出し元によってストレージが割り当てられたモデルの数。 allocatedProjectPathLen : プロジェクト パスを受け取るために 割り当てられたストレージの長さ。 allocatedModelPathLen : モデル パスを受け取るために 割り当てられたストレージの長さ。
出力パラメータ
numProjects : API がアクティブなプロジェクトの実際の数を格納する場所へのポインター。 numModels: API がすべてのアクティブなプロジェクトのモデルの実際の合計数を格納する場所へのポインター。 maxProjectPathLen: API が最長のプロジェクト パスの長さを格納する場所へのポインター。 maxModelPathLen: API が最長のモデル パスの長さを格納する場所へのポインター。 maxMachineNameLen: API が最長のマシン名の長さを格納する場所へのポインター。 maxIpLen: API が最長の IP アドレスの長さを格納する場所へのポインター。 projectPaths: API がアクティブなプロジェクトの構成ファイルのパスを格納する文字列の配列。 instanceIDs: API がアクティブなプロジェクトのインスタンス ID を格納する配列。 machineNames: API がマシン名を格納する配列。 controllerIPs: API が IP アドレスを格納する配列。 modelPaths: API がすべてのアクティブなプロジェクトのモデル パスを格納する配列。 numModelsPerProject: API が各プロジェクトのモデル数を格納する配列。これらの値を使用して、どのモデルがどのプロジェクトに属しているかを判断します。 modelIDs: API がすべてのアクティブなプロジェクトのモデル ID を格納する配列。 modelStates: API がすべてのアクティブなプロジェクトのモデルの状態を格納する配列。
また、OpalGetActiveProjects() 関数を使用して projectPaths、machineNames などを取得する C プログラムもあります。
#include <stdio.h>
#include <stdlib.h>
#include <malloc.h>
#include <string.h>
// Header for getcwd() is platform-dependent
#if defined(WIN32)
#include <direct.h>
#include <windows.h>
#elif defined(__linux__)
#include <unistd.h>
#define _MAX_PATH 256
#endif
#include "OpalApi.h"
void PrintError(int rc, char *funcName);
#define FALSE 0
#define TRUE 1
int main (int argc, char* argv[])
{
char **projectPath, **modelPath;
char *projectPaths, *modelPaths;
char **loaderMachineName, **controllerIp;
char *loaderMachineNames, *controllerIps;
int i, rc, projectIdx;
unsigned short *instId;
int *instId2;
int * modelIds;
unsigned short *numModelsPerProject;
unsigned short *modelId;
unsigned short *modelState;
unsigned short allocProjects, numProjects;
unsigned short allocModels, numModels;
unsigned short allocProjectPathLen, maxProjectPathLen;
unsigned short allocModelPathLen, maxModelPathLen;
unsigned short allocMachineLen, maxMachineLen;
unsigned short allocIpLen, maxIpLen;
// Obtenir la taille des données.
allocProjects =allocModels = 0;
allocProjectPathLen = allocModelPathLen = 0;
allocMachineLen = allocIpLen = 0;
rc = OpalGetActiveProjects( 0, &allocProjects,
0, &allocModels,
0, &allocProjectPathLen,
0, &allocModelPathLen,
0, &allocMachineLen,
0, &allocIpLen,
NULL, NULL, NULL, NULL,
NULL, NULL, NULL, NULL );
if ((rc != 0) && (rc != E2BIG))
{
PrintError(rc, "OpalGetActiveProject");
printf(" - alGetActiveProjects error output !!!\n");
printf("\t numProjects - %i \n", allocProjects);
printf("\t numModels - %i \n", allocModels);
printf("\t maxProjectPathLen - %i \n", allocProjectPathLen);
printf("\t maxModelPathLen - %i \n", allocModelPathLen);
//printf("\t maxMachineNameLen - %i \n", maxMachineNameLen);
printf("\t maxIpLen - %i \n", allocIpLen);
return (rc);
// Erreur
//return returnInstance;
}
projectPath = (char **)_alloca(allocProjects * sizeof(*projectPath));
projectPaths = (char *)_alloca(allocProjects * (allocProjectPathLen + 1) * sizeof(*projectPaths));
loaderMachineName = (char **)_alloca(allocProjects * sizeof(*loaderMachineName));
loaderMachineNames = (char *)_alloca(allocProjects * (allocMachineLen + 1) * sizeof(*loaderMachineNames));
controllerIp = (char **)_alloca(allocProjects * sizeof(*controllerIp));
controllerIps = (char *)_alloca(allocProjects * (allocIpLen + 1) * sizeof(*controllerIps));
numModelsPerProject = (unsigned short*)_alloca(allocProjects * sizeof(*numModelsPerProject));
modelPath = (char **)_alloca(allocModels * sizeof(*modelPath));
modelPaths = (char *)_alloca(allocModels * (allocModelPathLen + 1) * sizeof(*modelPaths));
for (i = 0; i < allocProjects; ++i)
{
projectPath[i] = &projectPaths[i * (allocProjectPathLen + 1)];
loaderMachineName[i] = &loaderMachineNames[i * (allocMachineLen + 1)];
controllerIp[i] = &controllerIps[i * (allocIpLen + 1)];
}
for (i = 0; i < allocModels; ++i)
{
modelPath[i] = &modelPaths[i * (allocModelPathLen + 1)];
}
instId = (unsigned short *)_alloca(allocProjects * sizeof(*instId));
instId2 = (int *)_alloca(allocProjects * sizeof(*instId2));
modelId = (unsigned short*)_alloca(allocModels * sizeof(*modelId));
modelState = (unsigned short *)_alloca(allocModels * sizeof(*modelState));
rc = OpalGetActiveProjects( allocProjects, &numProjects,
allocModels, &numModels,
allocProjectPathLen, &maxProjectPathLen,
allocModelPathLen, &maxModelPathLen,
allocMachineLen, &maxMachineLen,
allocIpLen, &maxIpLen,
projectPath, instId,
loaderMachineName, controllerIp, modelPath,
numModelsPerProject, modelId, modelState);
printf(" - OpalGetActiveProjects output !!!\n");
printf("\t numProjects - %i \n", allocProjects);
printf("\t numModels - %i \n", allocModels);
printf("\t maxProjectPathLen - %i \n", allocProjectPathLen);
printf("\t maxModelPathLen - %i \n", allocModelPathLen);
//printf("\t maxMachineNameLen - %i \n", maxMachineNameLen);
printf("\t maxIpLen - %i \n", allocIpLen);
for(i=0; i<numProjects; i++)
{
printf("\t \t projectPath[%d] = %s \n", i, *(projectPath +i));
printf("\t \t instId[%d] = %d \n", i, *(instId + i));
printf("\t \t loaderMachineName[%d] = %s \n", i, *(loaderMachineName + i));
printf("\t \t controllerIp[%d] = %s \n", i, *(controllerIp + i));
printf("\t \t numModelsPerProject[%d] = %d \n", i, * (numModelsPerProject +i));
}
for(i=0; i<numModels; i++)
{
printf("\t \t modelPath[%d] = %s \n", i, *(modelPath+i));
printf("\t \t modelId[%d] = %d \n", i, *(modelId +i));
printf("\t \t modelState[%d] = %d \n", i, *(modelState+i));
}
OpalDisconnect();
getchar();
}
void PrintError(int rc, char *funcName)
{
char buf[512];
unsigned short len;
OpalGetLastErrMsg(buf, sizeof(buf), &len);
printf("Error !!! \n %s (code %d) from %s\n", buf, rc, funcName);
}