gpt4 book ai didi

c# - SetFilePointer 不会失败,也不会移动指针

转载 作者:塔克拉玛干 更新时间:2023-11-03 00:40:46 25 4
gpt4 key购买 nike

首先,如果我说了一些愚蠢的话,请原谅我,我不是 IT 人员,而是电子工程师,但我被分配到需要更多技能的工作。

我需要写入和读取 SD 卡中的物理扇区。我已经用 C++ 完成了它,但主要应用程序是用 C# 编写的,所以我认为现在是编写我的第一个 dll 的好时机。

这里是用 C++ 写一个扇区的代码。

private: System::Void button4_Click(System::Object^  sender, System::EventArgs^  e) {
HANDLE hFile = INVALID_HANDLE_VALUE;
BOOL fSuccess = FALSE;


DWORD dwBytesWritten = 0;

unsigned char chBuffer[SIZE];

long SectorActual = 39;
long PosicionInicio = SectorActual * 512;

for (int i=0; i<SIZE; i++) // Garbage values to be written
{
chBuffer[i]= i % 16;
if((i/16)%2==0)
{
chBuffer[i]= 15 - chBuffer[i];
}
}
hFile = CreateFileA("\\\\.\\PhysicalDrive5",GENERIC_READ|GENERIC_WRITE,0,NULL,OPEN_EXISTING,FILE_ATTRIBUTE_NORMAL,NULL);

if (hFile == INVALID_HANDLE_VALUE)
{
textBox1->Text += "Could not open file (error " + GetLastError() + ") \r\n";
return;
}

DWORD dwPtr = SetFilePointer( hFile,
PosicionInicio,
NULL,
FILE_BEGIN );

if (dwPtr == INVALID_SET_FILE_POINTER) // Test for failure
{
textBox1->Text += "Error moving pointer (error " + GetLastError() + ") \r\n";
return; // Deal with failure
} // End of error handler

fSuccess = WriteFile(hFile, chBuffer, TAMANYO_SECTOR, &dwBytesWritten, NULL);
if (!fSuccess)
{
textBox1->Text += "WriteFile failed\r\n";
}
else
{
textBox1->Text += "WriteFile wrote " + dwBytesWritten.ToString() + " bytes\r\n";
}

CloseHandle(hFile);
}

然后我制作了DLL。功能是这样的:

 int AccesoBajoNivel::EscribeBloqueFisico(char numeroDisco, unsigned char * buffer, long      BytesQueEscribir, long posicion_inicio)
{
HANDLE hFile = INVALID_HANDLE_VALUE;
BOOL fSuccess = FALSE;
DWORD dwBytesWritten = 0;

char ArrayDeChars[] = "\\\\.\\PhysicalDrive5";
ArrayDeChars[17] = numeroDisco;
LPCSTR pathAlDisco = ArrayDeChars;

hFile = CreateFileA(pathAlDisco,GENERIC_READ|GENERIC_WRITE,0,NULL,OPEN_EXISTING,FILE_ATTRIBUTE_NORMAL,NULL);

if (hFile == INVALID_HANDLE_VALUE)
{
//printf("Could not open file (error %d)\n", GetLastError());
CloseHandle(hFile);
return -1;
}

DWORD dwPtr = SetFilePointer( hFile,
posicion_inicio,
NULL,
FILE_BEGIN );

if (dwPtr == INVALID_SET_FILE_POINTER) // Test for failure
{
CloseHandle(hFile);
return -2; // Deal with failure
} // End of error handler

fSuccess = WriteFile(hFile, buffer, BytesQueEscribir, &dwBytesWritten, NULL);
if (!fSuccess)
{
CloseHandle(hFile);
return -3;
}
else
{
CloseHandle(hFile);
//return dwBytesWritten;
return dwPtr;
}

CloseHandle(hFile);
}

然后我在C#项目中导入:

[DllImport("AccesoBajoNivel.dll", CallingConvention = CallingConvention.Cdecl)]
static extern int EscribeBloqueFisico(char numeroDisco, byte[] buffer, long BytesQueEscribir, long posicion_inicio);

然后我调用函数:

int ResultEscribir = EscribeBloqueFisico(numeroDiscoFisico, BufferEscritura, 512, SectorEscribir * 512);

总是写入第一个扇区,无论“posicion_inicio”的值如何(是开始偏移量,对西类牙标签感到抱歉)所以这就是为什么我更改了 api 中的写入函数以返回 dwPtr(SetFilePointer 的结果)写入的字节数。这似乎总是为零,这显然不是我想要的。但无法找到为什么第一个函数有效而 dll 调用无效。

也许是乍一看应该看到的东西,但正如我所说,我是电子专业人士,不习惯这种编程......

其他功能(如读取 DISK_GEOMETRY 或 VOLUME_DISK_EXTENTS)起作用,我用它来确定哪个物理驱动器是给定的逻辑单元。正如我所说,写作和阅读也很好,只是我总是指向零扇区......

提前致谢。另外,这是我第一次在这里发帖。看了很多遍(这就是为什么我选择在这里问)但是如果我在发布问题时犯了错误,请指出。

最佳答案

C++ 中的 long 是 32 位。这使得它成为 C# 中的 int。在 pinvoke 声明中将 long 替换为 int。

您应该已经收到关于此的 PInvokeStackImbalance 警告。如果您将其关闭,请确保重新启用该警告。很容易看出 *posicion_inicio* 的值在调试器中是错误的。请务必启用非托管调试、项目 + 属性、调试选项卡。现在您可以在 C++ 代码中设置断点。

关于c# - SetFilePointer 不会失败,也不会移动指针,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/12949937/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com