-
Notifications
You must be signed in to change notification settings - Fork 1.7k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Added WinioctlUtil, Ntifs and modified Kernel32 #732
Added WinioctlUtil, Ntifs and modified Kernel32 #732
Conversation
* Added `com.sun.jna.platform.win32.WinioctlUtil` for help in determining FSCTL_* codes * Added `com.sun.jna.platform.win32.Ntifs` with Reparse Point structures and defines * Added initialization of FILETIME from LARGE_INTEGER * Added `GetFileInformationByHandleEx` and `SetFileInformationByHandle` to `com.sun.jna.platform.win32.Kernel32`
Thank you - this looks very good. Comments:
|
@amarcionek I just remembered, that you asked on the jna-user mailing list about mapping of LPTSTR. Please review the structures and function mappings you introduced. Kernel32 is mapped like this: Kernel32 INSTANCE = Native.loadLibrary("kernel32", Kernel32.class, W32APIOptions.DEFAULT_OPTIONS); DEFAULT_OPTIONS is set dependend on the system property
This needs to be addressed you mapped: HANDLE FindFirstFile(LPWSTR /*MSDN: LPCTSTR*/ lpFileName, Pointer lpFindFileData); So I'd map this as: HANDLE FindFirstFile(String lpFileName, Pointer lpFindFileData); else the moment someone sets w32.ascii to true this blows. For the structures please also check which mapping is right a TCHAR depends on how it was created, if called from a unicode function it will be a WCHAR and from an ascii function it will be CHAR. Map the structure members to String and pass in the correct typemapper: W32APITypeMapper.DEFAULT I had to go through a pile of wrong structure mappings here: If I remember correctly these were also mixed. some structures contained LPTSTR (calling dependend string), some LPWSTR (always unicode). |
Thanks for reviewing my PR. I'm working on some of the items now. One comment on the Privilege class: You are correct that the thread token is being reset, in both the error case in enable() AND in the disable() function. We have used this class to enable a singular privilege for a short duration for a 1 or 2 function calls as needed. The idea is to enable the privilege, do your work, and then disable it immediately after. If you need multiple Privileges, then you'd need a class instance for each one. If you need multiple for a short duration and one fails, then arguably you can't complete the work within that block. You are correct it doesn't take into account impersonating in a code block elsewhere where the thread's token is already set. It was really designed to help remove the boilerplate. For multiples, I could see adding functionality to support more than one privilege and have them all on or off. I'd be happy to do that, or simply remove it and add that boilerplate into the test that needs it. |
Thanks - one datatype adjustment is needed UCHAR can't be mapped to char, but needs to be byte: https://msdn.microsoft.com/en-us/library/windows/desktop/aa383751(v=vs.85).aspx#UCHAR With regard to the Privilege class: I'm not that deep inside the privilege handling of windows - is it common or uncommon to need multiple privileges? If it is common I would see multiple privileges to be dealt with atomicly as a benefit. One other suggestion, as you pointed out, that you want to keep the privileged blocks small: Let Privilege implement Closable and map close to disable. If enable return the object itself this will become possible: Privilege privilegeSet = new Privilege("SE_INC_BASE_PRIORITY_NAME", "SE_INCREASE_QUOTA_NAME");
try(Privilege p1 = privilegeSet.enable()) {
// Do work with privilegeSet enabled
}
// privilegeSet disabled again With java 9 this should work: Privilege privilegeSet = new Privilege("SE_INC_BASE_PRIORITY_NAME", "SE_INCREASE_QUOTA_NAME");
try(privilegeSet.enable()) {
// Do work with privilegeSet enabled
}
// privilegeSet disabled again |
* Converted char to byte for UCHAR * Made Privilege implement Closeable
I believe the only UCHAR that needed to be changed from char to byte was the one in the compression structure. For Privilege, I implemented the ability to set multiple ones via a passed in String array. |
public class Privilege {
//...
public Privilege enable() {
// current implementation of enable
return this;
}
//...
}
try(Privilege enabledPrivilege = privilege.enable()) {
// do something
} |
* Changed Privilege constructor * Changed enable interface to return this * Removed try with resources * Fixed javadoc
I've made those changes. But I did run into one issue: the initial privilege gets a compiler warning if used in a try with resources: Resource leak: 'privilege' is never closed Code:
Even though we know privilegeEnabled is the same one, the compiler doesn't. Now, for JDK 7 we can't use try with resources, so for the unit test I added an explicit close in the finally block. |
@amarcionek Thank you Adam! I just merged the changeset into master. Please check this suggested change. This turns the WinioctlUtil members into final static ints. These can be "import static`ed". WinioctlFunction looks like an interface in search of a function (:-)) diff --git a/contrib/platform/src/com/sun/jna/platform/win32/WinioctlUtil.java b/contrib/platform/src/com/sun/jna/platform/win32/WinioctlUtil.java
index 5475dde..d92d49f 100644
--- a/contrib/platform/src/com/sun/jna/platform/win32/WinioctlUtil.java
+++ b/contrib/platform/src/com/sun/jna/platform/win32/WinioctlUtil.java
@@ -43,43 +43,33 @@
return ((DeviceType) << 16) | ((Access) << 14) | ((Function) << 2) | (Method);
}
- /**
- * Base interface for a Winiotcl function used to construct the control code
- */
- public interface WinioctlFunction {
- /**
- * @return the control code given the IOTCL's parameters
- */
- public abstract int getControlCode();
- }
+ public static final int FSCTL_GET_COMPRESSION = CTL_CODE(
+ Winioctl.FILE_DEVICE_FILE_SYSTEM,
+ Winioctl.FSCTL_GET_COMPRESSION,
+ Winioctl.METHOD_BUFFERED,
+ Winioctl.FILE_ANY_ACCESS);
- public static class FSCTL_GET_COMPRESSION implements WinioctlFunction {
- public int getControlCode() {
- return WinioctlUtil.CTL_CODE(Winioctl.FILE_DEVICE_FILE_SYSTEM, Winioctl.FSCTL_GET_COMPRESSION, Winioctl.METHOD_BUFFERED, Winioctl.FILE_ANY_ACCESS);
- }
- }
+ public static final int FSCTL_SET_COMPRESSION = CTL_CODE(
+ Winioctl.FILE_DEVICE_FILE_SYSTEM,
+ Winioctl.FSCTL_SET_COMPRESSION,
+ Winioctl.METHOD_BUFFERED,
+ WinNT.FILE_READ_DATA | WinNT.FILE_WRITE_DATA);
- public static class FSCTL_SET_COMPRESSION implements WinioctlFunction {
- public int getControlCode() {
- return WinioctlUtil.CTL_CODE(Winioctl.FILE_DEVICE_FILE_SYSTEM, Winioctl.FSCTL_SET_COMPRESSION, Winioctl.METHOD_BUFFERED, WinNT.FILE_READ_DATA | WinNT.FILE_WRITE_DATA);
- }
- }
+ public static final int FSCTL_SET_REPARSE_POINT = CTL_CODE(
+ Winioctl.FILE_DEVICE_FILE_SYSTEM,
+ Winioctl.FSCTL_SET_REPARSE_POINT,
+ Winioctl.METHOD_BUFFERED,
+ Winioctl.FILE_SPECIAL_ACCESS);
- public static class FSCTL_SET_REPARSE_POINT implements WinioctlFunction {
- public int getControlCode() {
- return WinioctlUtil.CTL_CODE(Winioctl.FILE_DEVICE_FILE_SYSTEM, Winioctl.FSCTL_SET_REPARSE_POINT, Winioctl.METHOD_BUFFERED, Winioctl.FILE_SPECIAL_ACCESS);
- }
- }
+ public static final int FSCTL_GET_REPARSE_POINT = CTL_CODE(
+ Winioctl.FILE_DEVICE_FILE_SYSTEM,
+ Winioctl.FSCTL_GET_REPARSE_POINT,
+ Winioctl.METHOD_BUFFERED,
+ Winioctl.FILE_ANY_ACCESS);
- public static class FSCTL_GET_REPARSE_POINT implements WinioctlFunction {
- public int getControlCode() {
- return WinioctlUtil.CTL_CODE(Winioctl.FILE_DEVICE_FILE_SYSTEM, Winioctl.FSCTL_GET_REPARSE_POINT, Winioctl.METHOD_BUFFERED, Winioctl.FILE_ANY_ACCESS);
- }
- }
-
- public static class FSCTL_DELETE_REPARSE_POINT implements WinioctlFunction {
- public int getControlCode() {
- return WinioctlUtil.CTL_CODE(Winioctl.FILE_DEVICE_FILE_SYSTEM, Winioctl.FSCTL_DELETE_REPARSE_POINT, Winioctl.METHOD_BUFFERED, Winioctl.FILE_SPECIAL_ACCESS);
- }
- }
+ public static final int FSCTL_DELETE_REPARSE_POINT = CTL_CODE(
+ Winioctl.FILE_DEVICE_FILE_SYSTEM,
+ Winioctl.FSCTL_DELETE_REPARSE_POINT,
+ Winioctl.METHOD_BUFFERED,
+ Winioctl.FILE_SPECIAL_ACCESS);
}
\ No newline at end of file
diff --git a/contrib/platform/test/com/sun/jna/platform/win32/Kernel32Test.java b/contrib/platform/test/com/sun/jna/platform/win32/Kernel32Test.java
index b0ca34c..5ae6720 100644
--- a/contrib/platform/test/com/sun/jna/platform/win32/Kernel32Test.java
+++ b/contrib/platform/test/com/sun/jna/platform/win32/Kernel32Test.java
@@ -62,10 +62,10 @@
import com.sun.jna.platform.win32.WinNT.MEMORY_BASIC_INFORMATION;
import com.sun.jna.platform.win32.WinNT.OSVERSIONINFO;
import com.sun.jna.platform.win32.WinNT.OSVERSIONINFOEX;
-import com.sun.jna.platform.win32.WinioctlUtil.FSCTL_GET_COMPRESSION;
-import com.sun.jna.platform.win32.WinioctlUtil.FSCTL_GET_REPARSE_POINT;
-import com.sun.jna.platform.win32.WinioctlUtil.FSCTL_SET_COMPRESSION;
-import com.sun.jna.platform.win32.WinioctlUtil.FSCTL_SET_REPARSE_POINT;
+import static com.sun.jna.platform.win32.WinioctlUtil.FSCTL_GET_COMPRESSION;
+import static com.sun.jna.platform.win32.WinioctlUtil.FSCTL_GET_REPARSE_POINT;
+import static com.sun.jna.platform.win32.WinioctlUtil.FSCTL_SET_COMPRESSION;
+import static com.sun.jna.platform.win32.WinioctlUtil.FSCTL_SET_REPARSE_POINT;
import com.sun.jna.ptr.IntByReference;
import com.sun.jna.ptr.ShortByReference;
@@ -587,7 +587,7 @@
IntByReference lpBytes = new IntByReference();
if (false == Kernel32.INSTANCE.DeviceIoControl(hFile,
- new FSCTL_GET_COMPRESSION().getControlCode(),
+ FSCTL_GET_COMPRESSION,
null,
0,
lpBuffer.getPointer(),
@@ -602,7 +602,7 @@
lpBuffer = new ShortByReference((short)WinNT.COMPRESSION_FORMAT_LZNT1);
if (false == Kernel32.INSTANCE.DeviceIoControl(hFile,
- new FSCTL_SET_COMPRESSION().getControlCode(),
+ FSCTL_SET_COMPRESSION,
lpBuffer.getPointer(),
USHORT.SIZE,
null,
@@ -613,7 +613,7 @@
}
if (false == Kernel32.INSTANCE.DeviceIoControl(hFile,
- new FSCTL_GET_COMPRESSION().getControlCode(),
+ FSCTL_GET_COMPRESSION,
null,
0,
lpBuffer.getPointer(),
@@ -666,7 +666,7 @@
REPARSE_DATA_BUFFER lpBuffer = new REPARSE_DATA_BUFFER(WinNT.IO_REPARSE_TAG_SYMLINK, (short) 0, symLinkReparseBuffer);
assertTrue(Kernel32.INSTANCE.DeviceIoControl(hFile,
- new FSCTL_SET_REPARSE_POINT().getControlCode(),
+ FSCTL_SET_REPARSE_POINT,
lpBuffer.getPointer(),
lpBuffer.getSize(),
null,
@@ -677,7 +677,7 @@
Memory p = new Memory(REPARSE_DATA_BUFFER.sizeOf());
IntByReference lpBytes = new IntByReference();
assertTrue(Kernel32.INSTANCE.DeviceIoControl(hFile,
- new FSCTL_GET_REPARSE_POINT().getControlCode(),
+ FSCTL_GET_REPARSE_POINT,
null,
0,
p, |
Thanks! I will get on that now. You are absolutely correct. |
…va-native-access#732) Motivation: The ByteBuf that is passed into the `writeToken(...)` method should use `BIG_ENDIAN` while at the moment it might use `LITTLE_ENDIAN`. Modifications: * Ensures that all buffers passed to the `QuicTokenHandler` are now big endian * Expose max connection it via static field in Quic class. Result: Fixes java-native-access#731. --------- Co-authored-by: Norman Maurer <[email protected]>
com.sun.jna.platform.win32.WinioctlUtil
for help in determining FSCTL_* codescom.sun.jna.platform.win32.Ntifs
with Reparse Point structures and definesGetFileInformationByHandleEx
andSetFileInformationByHandle
tocom.sun.jna.platform.win32.Kernel32