SetFileTime返回错误代码5

时间:2015-05-12 01:54:41

标签: c# file io pinvoke kernel32

我正在尝试P / Invoke SetFileTime,但我似乎无法让它工作。它总是为我返回错误代码5(访问被拒绝),我不知道为什么。这是我正在使用的代码:

void Main()
{
    var pointer = CreateFile(@"C:\Users\User\Desktop\New folder\New Text Document.txt", FileAccess.ReadWrite, FileShare.None, IntPtr.Zero, FileMode.Open, FileAttributes.Normal, IntPtr.Zero);
    Console.WriteLine(pointer);
    var now = DateTime.Now.ToFileTime();
    long lpCreationTime = now;
    long lpLastAccessTime = now;
    long lpLastWriteTime = now;
    if (!SetFileTime(pointer, ref lpCreationTime, ref lpLastAccessTime, ref lpLastWriteTime))
        Console.WriteLine(GetLastError());
    CloseHandle(pointer);
}

[DllImport("kernel32.dll")]
static extern UInt32 GetLastError();

[DllImport("kernel32.dll", SetLastError = true)]
static extern Boolean SetFileTime(IntPtr hFile, ref long lpCreationTime, ref long lpLastAccessTime, ref long lpLastWriteTime);

[DllImport("kernel32.dll", SetLastError = true)]
static extern Boolean CloseHandle(IntPtr hObject);

[DllImport("kernel32.dll", CharSet = CharSet.Unicode, SetLastError = true)]
static extern IntPtr CreateFile(String fileName, [MarshalAs(UnmanagedType.U4)] FileAccess fileAccess, [MarshalAs(UnmanagedType.U4)] FileShare fileShare, IntPtr securityAttributes, [MarshalAs(UnmanagedType.U4)] FileMode creationDisposition, [MarshalAs(UnmanagedType.U4)] FileAttributes flags, IntPtr template);

我的指针有效(2376)但我看到SetFileTime失败并返回错误代码5(拒绝访问)。现在,我确保我以管理员身份运行,并且我的帐户拥有该路径的权限,但仍然没有雪茄。任何人都有任何想法为什么会这样?

更新

调用Marshal.GetLastWin32Error()后,

SetFileTime也会返回5。此外,我需要让这个调用工作,以便SetFileTimeCreateFile支持的Windows中的长路径上,但.NET的当前文件库不支持Windows中的长路径。 / p>

2 个答案:

答案 0 :(得分:5)

来自SetFileTime的文档:

  

文件或目录的句柄。必须使用具有FILE_WRITE_ATTRIBUTES访问权限的CreateFile函数创建句柄。

您的代码无法做到这一点。 .net FileAccess枚举与Win32访问标志不兼容。您需要定义一个专门用于CreateFile的枚举。同样,您对FileShareFileMode的使用也不正确。

这个p / invoke声明应该足够了:http://www.pinvoke.net/default.aspx/kernel32.createfile

正如阿列克谢所说,不要调用GetLastError,因为您可能正在为框架调用而不是真正的错误找到错误代码。使用SetLastError = trueMarshal.GetLastWin32Error()

如果CreateFile,您也无法检查返回值中的错误。

答案 1 :(得分:1)

感谢所有帮助,我能够使用以下内容实现目标(FileAccess.ReadWrite转换为0x3,而FILE_WRITE_ATTRIBUTES为0x100):

[DllImport("kernel32.dll", SetLastError = true)]
static extern Boolean SetFileTime(SafeFileHandle hFile, ref long lpCreationTime, ref long lpLastAccessTime, ref long lpLastWriteTime);

[DllImport("kernel32.dll", CharSet = CharSet.Unicode, SetLastError = true)]
static extern SafeFileHandle CreateFile(String fileName, uint fileAccess, [MarshalAs(UnmanagedType.U4)] FileShare fileShare, IntPtr securityAttributes, [MarshalAs(UnmanagedType.U4)] FileMode creationDisposition, [MarshalAs(UnmanagedType.U4)] FileAttributes flags, IntPtr template);

static void Main(string[] args)
{
    var handle = CreateFile(@"C:\Users\User\Desktop\FileTimeTest.txt", (uint)(0x3 | 0x100), FileShare.None, IntPtr.Zero, FileMode.Create, FileAttributes.Normal, IntPtr.Zero);
    if (!handle.IsInvalid)
    {
        using (var stream = new FileStream(handle, FileAccess.ReadWrite))
        using (var writer = new StreamWriter(stream))
        {
            writer.WriteLine("Hello, world.");
            var now = DateTime.MaxValue.ToFileTime();
            if (!SetFileTime(handle, ref now, ref now, ref now))
                Console.WriteLine(Marshal.GetLastWin32Error());
        }
    }
}