diff --git a/src/hotspot/cpu/x86/x86.ad b/src/hotspot/cpu/x86/x86.ad
index 448e3c0a98d..023ef62ceea 100644
--- a/src/hotspot/cpu/x86/x86.ad
+++ b/src/hotspot/cpu/x86/x86.ad
@@ -1,5 +1,5 @@
//
-// Copyright (c) 2011, 2021, Oracle and/or its affiliates. All rights reserved.
+// Copyright (c) 2011, 2022, Oracle and/or its affiliates. All rights reserved.
// DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
//
// This code is free software; you can redistribute it and/or modify it
@@ -1783,17 +1783,9 @@ const bool Matcher::match_rule_supported_vector(int opcode, int vlen, BasicType
}
break;
case Op_VectorCastB2X:
- if (size_in_bits == 256 && UseAVX < 2) {
- return false; // Implementation limitation
- }
- break;
case Op_VectorCastS2X:
- if (is_integral_type(bt) && size_in_bits == 256 && UseAVX < 2) {
- return false;
- }
- break;
case Op_VectorCastI2X:
- if (is_integral_type(bt) && size_in_bits == 256 && UseAVX < 2) {
+ if (bt != T_DOUBLE && size_in_bits == 256 && UseAVX < 2) {
return false;
}
break;
@@ -6914,11 +6906,12 @@ instruct vcastBtoX(vec dst, vec src) %{
case T_LONG:
__ vpmovsxbq($dst$$XMMRegister, $src$$XMMRegister, vlen_enc);
break;
- case T_DOUBLE:
- __ vpmovsxbd($dst$$XMMRegister, $src$$XMMRegister, vlen_enc);
+ case T_DOUBLE: {
+ int mid_vlen_enc = (vlen_enc == Assembler::AVX_512bit) ? Assembler::AVX_256bit : Assembler::AVX_128bit;
+ __ vpmovsxbd($dst$$XMMRegister, $src$$XMMRegister, mid_vlen_enc);
__ vcvtdq2pd($dst$$XMMRegister, $dst$$XMMRegister, vlen_enc);
break;
-
+ }
default: assert(false, "%s", type2name(to_elem_bt));
}
%}
@@ -6985,10 +6978,12 @@ instruct vcastStoX_evex(vec dst, vec src) %{
case T_LONG:
__ vpmovsxwq($dst$$XMMRegister, $src$$XMMRegister, vlen_enc);
break;
- case T_DOUBLE:
- __ vpmovsxwd($dst$$XMMRegister, $src$$XMMRegister, vlen_enc);
+ case T_DOUBLE: {
+ int mid_vlen_enc = (vlen_enc == Assembler::AVX_512bit) ? Assembler::AVX_256bit : Assembler::AVX_128bit;
+ __ vpmovsxwd($dst$$XMMRegister, $src$$XMMRegister, mid_vlen_enc);
__ vcvtdq2pd($dst$$XMMRegister, $dst$$XMMRegister, vlen_enc);
break;
+ }
default:
ShouldNotReachHere();
}
diff --git a/src/java.base/share/classes/java/lang/invoke/MemoryAccessVarHandleBase.java b/src/java.base/share/classes/java/lang/invoke/MemoryAccessVarHandleBase.java
index 59098e9fa74..7703c997fc3 100644
--- a/src/java.base/share/classes/java/lang/invoke/MemoryAccessVarHandleBase.java
+++ b/src/java.base/share/classes/java/lang/invoke/MemoryAccessVarHandleBase.java
@@ -50,7 +50,7 @@ abstract class MemoryAccessVarHandleBase extends VarHandle {
this.alignmentMask = alignmentMask;
}
- static IllegalStateException newIllegalStateExceptionForMisalignedAccess(long address) {
- return new IllegalStateException("Misaligned access at address: " + address);
+ static IllegalArgumentException newIllegalArgumentExceptionForMisalignedAccess(long address) {
+ return new IllegalArgumentException("Misaligned access at address: " + address);
}
}
diff --git a/src/java.base/share/classes/java/lang/invoke/X-VarHandleMemoryAccess.java.template b/src/java.base/share/classes/java/lang/invoke/X-VarHandleMemoryAccess.java.template
index 0bdfd212e07..bdb4904d994 100644
--- a/src/java.base/share/classes/java/lang/invoke/X-VarHandleMemoryAccess.java.template
+++ b/src/java.base/share/classes/java/lang/invoke/X-VarHandleMemoryAccess.java.template
@@ -106,7 +106,7 @@ final class MemoryAccessVarHandle$Type$Helper extends MemoryAccessVarHandleBase
static long offset(boolean skipAlignmentMaskCheck, MemorySegmentProxy bb, long offset, long alignmentMask) {
long address = offsetNoVMAlignCheck(skipAlignmentMaskCheck, bb, offset, alignmentMask);
if ((address & VM_ALIGN) != 0) {
- throw MemoryAccessVarHandleBase.newIllegalStateExceptionForMisalignedAccess(address);
+ throw MemoryAccessVarHandleBase.newIllegalArgumentExceptionForMisalignedAccess(address);
}
return address;
}
@@ -115,14 +115,15 @@ final class MemoryAccessVarHandle$Type$Helper extends MemoryAccessVarHandleBase
static long offsetNoVMAlignCheck(boolean skipAlignmentMaskCheck, MemorySegmentProxy bb, long offset, long alignmentMask) {
long base = bb.unsafeGetOffset();
long address = base + offset;
+ long maxAlignMask = bb.maxAlignMask();
if (skipAlignmentMaskCheck) {
//note: the offset portion has already been aligned-checked, by construction
- if ((base & alignmentMask) != 0) {
- throw MemoryAccessVarHandleBase.newIllegalStateExceptionForMisalignedAccess(address);
+ if (((base | maxAlignMask) & alignmentMask) != 0) {
+ throw MemoryAccessVarHandleBase.newIllegalArgumentExceptionForMisalignedAccess(address);
}
} else {
- if ((address & alignmentMask) != 0) {
- throw MemoryAccessVarHandleBase.newIllegalStateExceptionForMisalignedAccess(address);
+ if (((address | maxAlignMask) & alignmentMask) != 0) {
+ throw MemoryAccessVarHandleBase.newIllegalArgumentExceptionForMisalignedAccess(address);
}
}
return address;
diff --git a/src/java.base/share/classes/java/security/Provider.java b/src/java.base/share/classes/java/security/Provider.java
index c6c3d63c61f..c1c4bea90cd 100644
--- a/src/java.base/share/classes/java/security/Provider.java
+++ b/src/java.base/share/classes/java/security/Provider.java
@@ -1,5 +1,5 @@
/*
- * Copyright (c) 1996, 2021, Oracle and/or its affiliates. All rights reserved.
+ * Copyright (c) 1996, 2022, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
@@ -1149,7 +1149,7 @@ public abstract class Provider extends Properties {
switch (opType) {
case ADD:
// clean up old alias if present
- Service prevAliasService = legacyMap.get(aliasAlg);
+ Service prevAliasService = legacyMap.get(aliasKey);
if (prevAliasService != null) {
prevAliasService.removeAlias(aliasAlg);
}
diff --git a/src/java.base/share/classes/jdk/internal/access/foreign/MemorySegmentProxy.java b/src/java.base/share/classes/jdk/internal/access/foreign/MemorySegmentProxy.java
index a00b6516a55..068fcfa1a6b 100644
--- a/src/java.base/share/classes/jdk/internal/access/foreign/MemorySegmentProxy.java
+++ b/src/java.base/share/classes/jdk/internal/access/foreign/MemorySegmentProxy.java
@@ -44,6 +44,7 @@ public abstract class MemorySegmentProxy {
public abstract Object unsafeGetBase();
public abstract boolean isSmall();
public abstract ScopedMemoryAccess.Scope scope();
+ public abstract long maxAlignMask();
/* Helper functions for offset computations. These are required so that we can avoid issuing long opcodes
* (e.g. LMUL, LADD) when we're operating on 'small' segments (segments whose length can be expressed with an int).
diff --git a/src/java.desktop/macosx/native/libawt_lwawt/awt/a11y/CommonComponentAccessibility.h b/src/java.desktop/macosx/native/libawt_lwawt/awt/a11y/CommonComponentAccessibility.h
index 33ab5c5fd2e..be0169a064d 100644
--- a/src/java.desktop/macosx/native/libawt_lwawt/awt/a11y/CommonComponentAccessibility.h
+++ b/src/java.desktop/macosx/native/libawt_lwawt/awt/a11y/CommonComponentAccessibility.h
@@ -83,6 +83,7 @@
- (NSView* _Nonnull)view;
- (NSWindow* _Nonnull)window;
- (id _Nonnull)parent;
+- (CommonComponentAccessibility* _Nullable)typeSafeParent;
- (NSString* _Nonnull)javaRole;
- (BOOL)isMenu;
diff --git a/src/java.desktop/macosx/native/libawt_lwawt/awt/a11y/CommonComponentAccessibility.m b/src/java.desktop/macosx/native/libawt_lwawt/awt/a11y/CommonComponentAccessibility.m
index 69fc9908fd5..d53d0350abf 100644
--- a/src/java.desktop/macosx/native/libawt_lwawt/awt/a11y/CommonComponentAccessibility.m
+++ b/src/java.desktop/macosx/native/libawt_lwawt/awt/a11y/CommonComponentAccessibility.m
@@ -95,7 +95,11 @@ static jobject sAccessibilityClass = NULL;
return NO;
}
- return isChildSelected(env, ((CommonComponentAccessibility *)[self parent])->fAccessible, fIndex, fComponent);
+ CommonComponentAccessibility* parent = [self typeSafeParent];
+ if (parent != nil) {
+ return isChildSelected(env, parent->fAccessible, fIndex, fComponent);
+ }
+ return NO;
}
- (BOOL)isSelectable:(JNIEnv *)env
@@ -708,6 +712,15 @@ static jobject sAccessibilityClass = NULL;
return fParent;
}
+- (CommonComponentAccessibility *)typeSafeParent
+{
+ id parent = [self parent];
+ if ([parent isKindOfClass:[CommonComponentAccessibility class]]) {
+ return (CommonComponentAccessibility*)parent;
+ }
+ return nil;
+}
+
- (NSString *)javaRole
{
if(fJavaRole == nil) {
@@ -824,11 +837,13 @@ static jobject sAccessibilityClass = NULL;
if (fNSRole == nil) {
NSString *javaRole = [self javaRole];
fNSRole = [sRoles objectForKey:javaRole];
+ CommonComponentAccessibility* parent = [self typeSafeParent];
// The sRoles NSMutableDictionary maps popupmenu to Mac's popup button.
// JComboBox behavior currently relies on this. However this is not the
// proper mapping for a JPopupMenu so fix that.
if ( [javaRole isEqualToString:@"popupmenu"] &&
- ![[[self parent] javaRole] isEqualToString:@"combobox"] ) {
+ parent != nil &&
+ ![[parent javaRole] isEqualToString:@"combobox"] ) {
fNSRole = NSAccessibilityMenuRole;
}
if (fNSRole == nil) {
@@ -1025,8 +1040,9 @@ static jobject sAccessibilityClass = NULL;
// This may change when later fixing issues which currently
// exist for combo boxes, but for now the following is only
// for JPopupMenus, not for combobox menus.
- id parent = [self parent];
+ id parent = [self typeSafeParent];
if ( [[self javaRole] isEqualToString:@"popupmenu"] &&
+ parent != nil &&
![[parent javaRole] isEqualToString:@"combobox"] ) {
NSArray *children =
[CommonComponentAccessibility childrenOfParent:self
diff --git a/src/java.desktop/macosx/native/libawt_lwawt/awt/a11y/TabButtonAccessibility.m b/src/java.desktop/macosx/native/libawt_lwawt/awt/a11y/TabButtonAccessibility.m
index 4caf97c21cb..538240028b8 100644
--- a/src/java.desktop/macosx/native/libawt_lwawt/awt/a11y/TabButtonAccessibility.m
+++ b/src/java.desktop/macosx/native/libawt_lwawt/awt/a11y/TabButtonAccessibility.m
@@ -59,11 +59,14 @@
- (jobject)tabGroup
{
if (fTabGroupAxContext == NULL) {
- JNIEnv* env = [ThreadUtilities getJNIEnv];
- jobject tabGroupAxContext = [(CommonComponentAccessibility *)[self parent] axContextWithEnv:env];
- fTabGroupAxContext = (*env)->NewWeakGlobalRef(env, tabGroupAxContext);
- CHECK_EXCEPTION();
- (*env)->DeleteLocalRef(env, tabGroupAxContext);
+ CommonComponentAccessibility* parent = [self typeSafeParent];
+ if (parent != nil) {
+ JNIEnv *env = [ThreadUtilities getJNIEnv];
+ jobject tabGroupAxContext = [parent axContextWithEnv:env];
+ fTabGroupAxContext = (*env)->NewWeakGlobalRef(env, tabGroupAxContext);
+ CHECK_EXCEPTION();
+ (*env)->DeleteLocalRef(env, tabGroupAxContext);
+ }
}
return fTabGroupAxContext;
}
diff --git a/src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/MemoryAddress.java b/src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/MemoryAddress.java
index 7e94d63014c..47a17799700 100644
--- a/src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/MemoryAddress.java
+++ b/src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/MemoryAddress.java
@@ -174,6 +174,8 @@ public sealed interface MemoryAddress extends Addressable permits MemoryAddressI
* @param layout the layout of the memory region to be read.
* @param offset offset in bytes (relative to this address). The final address of this read operation can be expressed as {@code toRowLongValue() + offset}.
* @return a byte value read from this address.
+ * @throws IllegalArgumentException if the dereference operation is
+ * incompatible with the alignment constraints in the provided layout.
* @throws IllegalCallerException if access to this method occurs from a module {@code M} and the command line option
* {@code --enable-native-access} is either absent, or does not mention the module name {@code M}, or
* {@code ALL-UNNAMED} in case {@code M} is an unnamed module.
@@ -192,6 +194,8 @@ public sealed interface MemoryAddress extends Addressable permits MemoryAddressI
* @param layout the layout of the memory region to be written.
* @param offset offset in bytes (relative to this address). The final address of this write operation can be expressed as {@code toRowLongValue() + offset}.
* @param value the byte value to be written.
+ * @throws IllegalArgumentException if the dereference operation is
+ * incompatible with the alignment constraints in the provided layout.
* @throws IllegalCallerException if access to this method occurs from a module {@code M} and the command line option
* {@code --enable-native-access} is either absent, or does not mention the module name {@code M}, or
* {@code ALL-UNNAMED} in case {@code M} is an unnamed module.
@@ -210,6 +214,8 @@ public sealed interface MemoryAddress extends Addressable permits MemoryAddressI
* @param layout the layout of the memory region to be read.
* @param offset offset in bytes (relative to this address). The final address of this read operation can be expressed as {@code toRowLongValue() + offset}.
* @return a boolean value read from this address.
+ * @throws IllegalArgumentException if the dereference operation is
+ * incompatible with the alignment constraints in the provided layout.
* @throws IllegalCallerException if access to this method occurs from a module {@code M} and the command line option
* {@code --enable-native-access} is either absent, or does not mention the module name {@code M}, or
* {@code ALL-UNNAMED} in case {@code M} is an unnamed module.
@@ -228,6 +234,8 @@ public sealed interface MemoryAddress extends Addressable permits MemoryAddressI
* @param layout the layout of the memory region to be written.
* @param offset offset in bytes (relative to this address). The final address of this write operation can be expressed as {@code toRowLongValue() + offset}.
* @param value the boolean value to be written.
+ * @throws IllegalArgumentException if the dereference operation is
+ * incompatible with the alignment constraints in the provided layout.
* @throws IllegalCallerException if access to this method occurs from a module {@code M} and the command line option
* {@code --enable-native-access} is either absent, or does not mention the module name {@code M}, or
* {@code ALL-UNNAMED} in case {@code M} is an unnamed module.
@@ -246,6 +254,8 @@ public sealed interface MemoryAddress extends Addressable permits MemoryAddressI
* @param layout the layout of the memory region to be read.
* @param offset offset in bytes (relative to this address). The final address of this read operation can be expressed as {@code toRowLongValue() + offset}.
* @return a char value read from this address.
+ * @throws IllegalArgumentException if the dereference operation is
+ * incompatible with the alignment constraints in the provided layout.
* @throws IllegalCallerException if access to this method occurs from a module {@code M} and the command line option
* {@code --enable-native-access} is either absent, or does not mention the module name {@code M}, or
* {@code ALL-UNNAMED} in case {@code M} is an unnamed module.
@@ -264,6 +274,8 @@ public sealed interface MemoryAddress extends Addressable permits MemoryAddressI
* @param layout the layout of the memory region to be written.
* @param offset offset in bytes (relative to this address). The final address of this write operation can be expressed as {@code toRowLongValue() + offset}.
* @param value the char value to be written.
+ * @throws IllegalArgumentException if the dereference operation is
+ * incompatible with the alignment constraints in the provided layout.
* @throws IllegalCallerException if access to this method occurs from a module {@code M} and the command line option
* {@code --enable-native-access} is either absent, or does not mention the module name {@code M}, or
* {@code ALL-UNNAMED} in case {@code M} is an unnamed module.
@@ -282,6 +294,8 @@ public sealed interface MemoryAddress extends Addressable permits MemoryAddressI
* @param layout the layout of the memory region to be read.
* @param offset offset in bytes (relative to this address). The final address of this read operation can be expressed as {@code toRowLongValue() + offset}.
* @return a short value read from this address.
+ * @throws IllegalArgumentException if the dereference operation is
+ * incompatible with the alignment constraints in the provided layout.
* @throws IllegalCallerException if access to this method occurs from a module {@code M} and the command line option
* {@code --enable-native-access} is either absent, or does not mention the module name {@code M}, or
* {@code ALL-UNNAMED} in case {@code M} is an unnamed module.
@@ -300,6 +314,8 @@ public sealed interface MemoryAddress extends Addressable permits MemoryAddressI
* @param layout the layout of the memory region to be written.
* @param offset offset in bytes (relative to this address). The final address of this write operation can be expressed as {@code toRowLongValue() + offset}.
* @param value the short value to be written.
+ * @throws IllegalArgumentException if the dereference operation is
+ * incompatible with the alignment constraints in the provided layout.
* @throws IllegalCallerException if access to this method occurs from a module {@code M} and the command line option
* {@code --enable-native-access} is either absent, or does not mention the module name {@code M}, or
* {@code ALL-UNNAMED} in case {@code M} is an unnamed module.
@@ -318,6 +334,8 @@ public sealed interface MemoryAddress extends Addressable permits MemoryAddressI
* @param layout the layout of the memory region to be read.
* @param offset offset in bytes (relative to this address). The final address of this read operation can be expressed as {@code toRowLongValue() + offset}.
* @return an int value read from this address.
+ * @throws IllegalArgumentException if the dereference operation is
+ * incompatible with the alignment constraints in the provided layout.
* @throws IllegalCallerException if access to this method occurs from a module {@code M} and the command line option
* {@code --enable-native-access} is either absent, or does not mention the module name {@code M}, or
* {@code ALL-UNNAMED} in case {@code M} is an unnamed module.
@@ -336,6 +354,8 @@ public sealed interface MemoryAddress extends Addressable permits MemoryAddressI
* @param layout the layout of the memory region to be written.
* @param offset offset in bytes (relative to this address). The final address of this write operation can be expressed as {@code toRowLongValue() + offset}.
* @param value the int value to be written.
+ * @throws IllegalArgumentException if the dereference operation is
+ * incompatible with the alignment constraints in the provided layout.
* @throws IllegalCallerException if access to this method occurs from a module {@code M} and the command line option
* {@code --enable-native-access} is either absent, or does not mention the module name {@code M}, or
* {@code ALL-UNNAMED} in case {@code M} is an unnamed module.
@@ -354,6 +374,8 @@ public sealed interface MemoryAddress extends Addressable permits MemoryAddressI
* @param layout the layout of the memory region to be read.
* @param offset offset in bytes (relative to this address). The final address of this read operation can be expressed as {@code toRowLongValue() + offset}.
* @return a float value read from this address.
+ * @throws IllegalArgumentException if the dereference operation is
+ * incompatible with the alignment constraints in the provided layout.
* @throws IllegalCallerException if access to this method occurs from a module {@code M} and the command line option
* {@code --enable-native-access} is either absent, or does not mention the module name {@code M}, or
* {@code ALL-UNNAMED} in case {@code M} is an unnamed module.
@@ -372,6 +394,8 @@ public sealed interface MemoryAddress extends Addressable permits MemoryAddressI
* @param layout the layout of the memory region to be written.
* @param offset offset in bytes (relative to this address). The final address of this write operation can be expressed as {@code toRowLongValue() + offset}.
* @param value the float value to be written.
+ * @throws IllegalArgumentException if the dereference operation is
+ * incompatible with the alignment constraints in the provided layout.
* @throws IllegalCallerException if access to this method occurs from a module {@code M} and the command line option
* {@code --enable-native-access} is either absent, or does not mention the module name {@code M}, or
* {@code ALL-UNNAMED} in case {@code M} is an unnamed module.
@@ -390,6 +414,8 @@ public sealed interface MemoryAddress extends Addressable permits MemoryAddressI
* @param layout the layout of the memory region to be read.
* @param offset offset in bytes (relative to this address). The final address of this read operation can be expressed as {@code toRowLongValue() + offset}.
* @return a long value read from this address.
+ * @throws IllegalArgumentException if the dereference operation is
+ * incompatible with the alignment constraints in the provided layout.
* @throws IllegalCallerException if access to this method occurs from a module {@code M} and the command line option
* {@code --enable-native-access} is either absent, or does not mention the module name {@code M}, or
* {@code ALL-UNNAMED} in case {@code M} is an unnamed module.
@@ -408,6 +434,8 @@ public sealed interface MemoryAddress extends Addressable permits MemoryAddressI
* @param layout the layout of the memory region to be written.
* @param offset offset in bytes (relative to this address). The final address of this write operation can be expressed as {@code toRowLongValue() + offset}.
* @param value the long value to be written.
+ * @throws IllegalArgumentException if the dereference operation is
+ * incompatible with the alignment constraints in the provided layout.
* @throws IllegalCallerException if access to this method occurs from a module {@code M} and the command line option
* {@code --enable-native-access} is either absent, or does not mention the module name {@code M}, or
* {@code ALL-UNNAMED} in case {@code M} is an unnamed module.
@@ -426,6 +454,8 @@ public sealed interface MemoryAddress extends Addressable permits MemoryAddressI
* @param layout the layout of the memory region to be read.
* @param offset offset in bytes (relative to this address). The final address of this read operation can be expressed as {@code toRowLongValue() + offset}.
* @return a double value read from this address.
+ * @throws IllegalArgumentException if the dereference operation is
+ * incompatible with the alignment constraints in the provided layout.
* @throws IllegalCallerException if access to this method occurs from a module {@code M} and the command line option
* {@code --enable-native-access} is either absent, or does not mention the module name {@code M}, or
* {@code ALL-UNNAMED} in case {@code M} is an unnamed module.
@@ -444,6 +474,8 @@ public sealed interface MemoryAddress extends Addressable permits MemoryAddressI
* @param layout the layout of the memory region to be written.
* @param offset offset in bytes (relative to this address). The final address of this write operation can be expressed as {@code toRowLongValue() + offset}.
* @param value the double value to be written.
+ * @throws IllegalArgumentException if the dereference operation is
+ * incompatible with the alignment constraints in the provided layout.
* @throws IllegalCallerException if access to this method occurs from a module {@code M} and the command line option
* {@code --enable-native-access} is either absent, or does not mention the module name {@code M}, or
* {@code ALL-UNNAMED} in case {@code M} is an unnamed module.
@@ -462,6 +494,8 @@ public sealed interface MemoryAddress extends Addressable permits MemoryAddressI
* @param layout the layout of the memory region to be read.
* @param offset offset in bytes (relative to this address). The final address of this read operation can be expressed as {@code toRowLongValue() + offset}.
* @return an address value read from this address.
+ * @throws IllegalArgumentException if the dereference operation is
+ * incompatible with the alignment constraints in the provided layout.
* @throws IllegalCallerException if access to this method occurs from a module {@code M} and the command line option
* {@code --enable-native-access} is either absent, or does not mention the module name {@code M}, or
* {@code ALL-UNNAMED} in case {@code M} is an unnamed module.
@@ -480,6 +514,8 @@ public sealed interface MemoryAddress extends Addressable permits MemoryAddressI
* @param layout the layout of the memory region to be written.
* @param offset offset in bytes (relative to this address). The final address of this write operation can be expressed as {@code toRowLongValue() + offset}.
* @param value the address value to be written.
+ * @throws IllegalArgumentException if the dereference operation is
+ * incompatible with the alignment constraints in the provided layout.
* @throws IllegalCallerException if access to this method occurs from a module {@code M} and the command line option
* {@code --enable-native-access} is either absent, or does not mention the module name {@code M}, or
* {@code ALL-UNNAMED} in case {@code M} is an unnamed module.
@@ -498,6 +534,9 @@ public sealed interface MemoryAddress extends Addressable permits MemoryAddressI
* @param layout the layout of the memory region to be read.
* @param index index in bytes (relative to this address). The final address of this read operation can be expressed as {@code toRowLongValue() + (index * layout.byteSize())}.
* @return a char value read from this address.
+ * @throws IllegalArgumentException if the dereference operation is
+ * incompatible with the alignment constraints in the provided layout,
+ * or if the layout alignment is greater than its size.
* @throws IllegalCallerException if access to this method occurs from a module {@code M} and the command line option
* {@code --enable-native-access} is either absent, or does not mention the module name {@code M}, or
* {@code ALL-UNNAMED} in case {@code M} is an unnamed module.
@@ -516,6 +555,9 @@ public sealed interface MemoryAddress extends Addressable permits MemoryAddressI
* @param layout the layout of the memory region to be written.
* @param index index in bytes (relative to this address). The final address of this write operation can be expressed as {@code toRowLongValue() + (index * layout.byteSize())}.
* @param value the char value to be written.
+ * @throws IllegalArgumentException if the dereference operation is
+ * incompatible with the alignment constraints in the provided layout,
+ * or if the layout alignment is greater than its size.
* @throws IllegalCallerException if access to this method occurs from a module {@code M} and the command line option
* {@code --enable-native-access} is either absent, or does not mention the module name {@code M}, or
* {@code ALL-UNNAMED} in case {@code M} is an unnamed module.
@@ -534,6 +576,9 @@ public sealed interface MemoryAddress extends Addressable permits MemoryAddressI
* @param layout the layout of the memory region to be read.
* @param index index in bytes (relative to this address). The final address of this read operation can be expressed as {@code toRowLongValue() + (index * layout.byteSize())}.
* @return a short value read from this address.
+ * @throws IllegalArgumentException if the dereference operation is
+ * incompatible with the alignment constraints in the provided layout,
+ * or if the layout alignment is greater than its size.
* @throws IllegalCallerException if access to this method occurs from a module {@code M} and the command line option
* {@code --enable-native-access} is either absent, or does not mention the module name {@code M}, or
* {@code ALL-UNNAMED} in case {@code M} is an unnamed module.
@@ -552,6 +597,9 @@ public sealed interface MemoryAddress extends Addressable permits MemoryAddressI
* @param layout the layout of the memory region to be written.
* @param index index in bytes (relative to this address). The final address of this write operation can be expressed as {@code toRowLongValue() + (index * layout.byteSize())}.
* @param value the short value to be written.
+ * @throws IllegalArgumentException if the dereference operation is
+ * incompatible with the alignment constraints in the provided layout,
+ * or if the layout alignment is greater than its size.
* @throws IllegalCallerException if access to this method occurs from a module {@code M} and the command line option
* {@code --enable-native-access} is either absent, or does not mention the module name {@code M}, or
* {@code ALL-UNNAMED} in case {@code M} is an unnamed module.
@@ -570,6 +618,9 @@ public sealed interface MemoryAddress extends Addressable permits MemoryAddressI
* @param layout the layout of the memory region to be read.
* @param index index in bytes (relative to this address). The final address of this read operation can be expressed as {@code toRowLongValue() + (index * layout.byteSize())}.
* @return an int value read from this address.
+ * @throws IllegalArgumentException if the dereference operation is
+ * incompatible with the alignment constraints in the provided layout,
+ * or if the layout alignment is greater than its size.
* @throws IllegalCallerException if access to this method occurs from a module {@code M} and the command line option
* {@code --enable-native-access} is either absent, or does not mention the module name {@code M}, or
* {@code ALL-UNNAMED} in case {@code M} is an unnamed module.
@@ -588,6 +639,9 @@ public sealed interface MemoryAddress extends Addressable permits MemoryAddressI
* @param layout the layout of the memory region to be written.
* @param index index in bytes (relative to this address). The final address of this write operation can be expressed as {@code toRowLongValue() + (index * layout.byteSize())}.
* @param value the int value to be written.
+ * @throws IllegalArgumentException if the dereference operation is
+ * incompatible with the alignment constraints in the provided layout,
+ * or if the layout alignment is greater than its size.
* @throws IllegalCallerException if access to this method occurs from a module {@code M} and the command line option
* {@code --enable-native-access} is either absent, or does not mention the module name {@code M}, or
* {@code ALL-UNNAMED} in case {@code M} is an unnamed module.
@@ -606,6 +660,9 @@ public sealed interface MemoryAddress extends Addressable permits MemoryAddressI
* @param layout the layout of the memory region to be read.
* @param index index in bytes (relative to this address). The final address of this read operation can be expressed as {@code toRowLongValue() + (index * layout.byteSize())}.
* @return a float value read from this address.
+ * @throws IllegalArgumentException if the dereference operation is
+ * incompatible with the alignment constraints in the provided layout,
+ * or if the layout alignment is greater than its size.
* @throws IllegalCallerException if access to this method occurs from a module {@code M} and the command line option
* {@code --enable-native-access} is either absent, or does not mention the module name {@code M}, or
* {@code ALL-UNNAMED} in case {@code M} is an unnamed module.
@@ -624,6 +681,9 @@ public sealed interface MemoryAddress extends Addressable permits MemoryAddressI
* @param layout the layout of the memory region to be written.
* @param index index in bytes (relative to this address). The final address of this write operation can be expressed as {@code toRowLongValue() + (index * layout.byteSize())}.
* @param value the float value to be written.
+ * @throws IllegalArgumentException if the dereference operation is
+ * incompatible with the alignment constraints in the provided layout,
+ * or if the layout alignment is greater than its size.
* @throws IllegalCallerException if access to this method occurs from a module {@code M} and the command line option
* {@code --enable-native-access} is either absent, or does not mention the module name {@code M}, or
* {@code ALL-UNNAMED} in case {@code M} is an unnamed module.
@@ -642,6 +702,9 @@ public sealed interface MemoryAddress extends Addressable permits MemoryAddressI
* @param layout the layout of the memory region to be read.
* @param index index in bytes (relative to this address). The final address of this read operation can be expressed as {@code toRowLongValue() + (index * layout.byteSize())}.
* @return a long value read from this address.
+ * @throws IllegalArgumentException if the dereference operation is
+ * incompatible with the alignment constraints in the provided layout,
+ * or if the layout alignment is greater than its size.
* @throws IllegalCallerException if access to this method occurs from a module {@code M} and the command line option
* {@code --enable-native-access} is either absent, or does not mention the module name {@code M}, or
* {@code ALL-UNNAMED} in case {@code M} is an unnamed module.
@@ -660,6 +723,9 @@ public sealed interface MemoryAddress extends Addressable permits MemoryAddressI
* @param layout the layout of the memory region to be written.
* @param index index in bytes (relative to this address). The final address of this write operation can be expressed as {@code toRowLongValue() + (index * layout.byteSize())}.
* @param value the long value to be written.
+ * @throws IllegalArgumentException if the dereference operation is
+ * incompatible with the alignment constraints in the provided layout,
+ * or if the layout alignment is greater than its size.
* @throws IllegalCallerException if access to this method occurs from a module {@code M} and the command line option
* {@code --enable-native-access} is either absent, or does not mention the module name {@code M}, or
* {@code ALL-UNNAMED} in case {@code M} is an unnamed module.
@@ -678,6 +744,9 @@ public sealed interface MemoryAddress extends Addressable permits MemoryAddressI
* @param layout the layout of the memory region to be read.
* @param index index in bytes (relative to this address). The final address of this read operation can be expressed as {@code toRowLongValue() + (index * layout.byteSize())}.
* @return a double value read from this address.
+ * @throws IllegalArgumentException if the dereference operation is
+ * incompatible with the alignment constraints in the provided layout,
+ * or if the layout alignment is greater than its size.
* @throws IllegalCallerException if access to this method occurs from a module {@code M} and the command line option
* {@code --enable-native-access} is either absent, or does not mention the module name {@code M}, or
* {@code ALL-UNNAMED} in case {@code M} is an unnamed module.
@@ -696,6 +765,9 @@ public sealed interface MemoryAddress extends Addressable permits MemoryAddressI
* @param layout the layout of the memory region to be written.
* @param index index in bytes (relative to this address). The final address of this write operation can be expressed as {@code toRowLongValue() + (index * layout.byteSize())}.
* @param value the double value to be written.
+ * @throws IllegalArgumentException if the dereference operation is
+ * incompatible with the alignment constraints in the provided layout,
+ * or if the layout alignment is greater than its size.
* @throws IllegalCallerException if access to this method occurs from a module {@code M} and the command line option
* {@code --enable-native-access} is either absent, or does not mention the module name {@code M}, or
* {@code ALL-UNNAMED} in case {@code M} is an unnamed module.
@@ -714,6 +786,9 @@ public sealed interface MemoryAddress extends Addressable permits MemoryAddressI
* @param layout the layout of the memory region to be read.
* @param index index in bytes (relative to this address). The final address of this read operation can be expressed as {@code toRowLongValue() + (index * layout.byteSize())}.
* @return an address value read from this address.
+ * @throws IllegalArgumentException if the dereference operation is
+ * incompatible with the alignment constraints in the provided layout,
+ * or if the layout alignment is greater than its size.
* @throws IllegalCallerException if access to this method occurs from a module {@code M} and the command line option
* {@code --enable-native-access} is either absent, or does not mention the module name {@code M}, or
* {@code ALL-UNNAMED} in case {@code M} is an unnamed module.
@@ -732,6 +807,9 @@ public sealed interface MemoryAddress extends Addressable permits MemoryAddressI
* @param layout the layout of the memory region to be written.
* @param index index in bytes (relative to this address). The final address of this write operation can be expressed as {@code toRowLongValue() + (index * layout.byteSize())}.
* @param value the address value to be written.
+ * @throws IllegalArgumentException if the dereference operation is
+ * incompatible with the alignment constraints in the provided layout,
+ * or if the layout alignment is greater than its size.
* @throws IllegalCallerException if access to this method occurs from a module {@code M} and the command line option
* {@code --enable-native-access} is either absent, or does not mention the module name {@code M}, or
* {@code ALL-UNNAMED} in case {@code M} is an unnamed module.
diff --git a/src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/MemorySegment.java b/src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/MemorySegment.java
index de974fa2e80..40cee4cf7d7 100644
--- a/src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/MemorySegment.java
+++ b/src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/MemorySegment.java
@@ -139,6 +139,56 @@ import java.util.stream.Stream;
* {@linkplain MemoryHandles#varHandle(ValueLayout) value layout}, and then adapt it using the var handle combinator
* functions defined in the {@link MemoryHandles} class.
*
+ *
Alignment
+ *
+ * When dereferencing a memory segment using a layout, the runtime must check that the segment address being dereferenced
+ * matches the layout's {@linkplain MemoryLayout#byteAlignment() alignment constraints}. If the segment being
+ * dereferenced is a native segment, then it has a concrete {@linkplain #address() base address}, which can
+ * be used to perform the alignment check. The pseudo-function below demonstrates this:
+ *
+ *
+ *
+ * If, however, the segment being dereferenced is a heap segment, the above function will not work: a heap
+ * segment's base address is virtualized and, as such, cannot be used to construct an alignment check. Instead,
+ * heap segments are assumed to produce addresses which are never more aligned than the element size of the Java array from which
+ * they have originated from, as shown in the following table:
+ *
+ *
+ *
Array type of an array backing a segment and its address alignment
+ *
+ *
+ *
Array type
+ *
Alignment
+ *
+ *
+ *
+ *
{@code boolean[]}
+ *
{@code 1}
+ *
{@code byte[]}
+ *
{@code 1}
+ *
{@code char[]}
+ *
{@code 2}
+ *
{@code short[]}
+ *
{@code 2}
+ *
{@code int[]}
+ *
{@code 4}
+ *
{@code float[]}
+ *
{@code 4}
+ *
{@code long[]}
+ *
{@code 8}
+ *
{@code double[]}
+ *
{@code 8}
+ *
+ *
+ *
+ * Note that the above definition is conservative: it might be possible, for instance, that a heap segment
+ * constructed from a {@code byte[]} might have a subset of addresses {@code S} which happen to be 8-byte aligned. But determining
+ * which segment addresses belong to {@code S} requires reasoning about details which are ultimately implementation-dependent.
+ *
*
Lifecycle and confinement
*
* Memory segments are associated with a resource scope (see {@link ResourceScope}), which can be accessed using
@@ -226,7 +276,9 @@ public sealed interface MemorySegment extends Addressable permits AbstractMemory
* @param elementLayout the layout to be used for splitting.
* @return the element spliterator for this segment
* @throws IllegalArgumentException if the {@code elementLayout} size is zero, or the segment size modulo the
- * {@code elementLayout} size is greater than zero.
+ * {@code elementLayout} size is greater than zero, if this segment is
+ * incompatible with the alignment constraints in the provided layout,
+ * or if the {@code elementLayout} alignment is greater than its size.
*/
Spliterator spliterator(MemoryLayout elementLayout);
@@ -240,7 +292,9 @@ public sealed interface MemorySegment extends Addressable permits AbstractMemory
* @param elementLayout the layout to be used for splitting.
* @return a sequential {@code Stream} over disjoint slices in this segment.
* @throws IllegalArgumentException if the {@code elementLayout} size is zero, or the segment size modulo the
- * {@code elementLayout} size is greater than zero.
+ * {@code elementLayout} size is greater than zero, if this segment is
+ * incompatible with the alignment constraints in the provided layout,
+ * or if the {@code elementLayout} alignment is greater than its size.
*/
Stream elements(MemoryLayout elementLayout);
@@ -982,9 +1036,9 @@ public sealed interface MemorySegment extends Addressable permits AbstractMemory
* @param dstElementLayout the element layout associated with the destination segment.
* @param dstOffset the starting offset, in bytes, of the destination segment.
* @param elementCount the number of elements to be copied.
- * @throws IllegalArgumentException if the element layouts have different sizes, if the source offset is incompatible
- * with the alignment constraints in the source element layout, or if the destination offset is incompatible with the
- * alignment constraints in the destination element layout.
+ * @throws IllegalArgumentException if the element layouts have different sizes, if the source (resp. destination) segment/offset are
+ * incompatible with the alignment constraints in the source
+ * (resp. destination) element layout, or if the source (resp. destination) element layout alignment is greater than its size.
* @throws IllegalStateException if either the scope associated with the source segment or the scope associated
* with the destination segment have been already closed, or if access occurs from a thread other than the thread
* owning either scopes.
@@ -1003,13 +1057,15 @@ public sealed interface MemorySegment extends Addressable permits AbstractMemory
AbstractMemorySegmentImpl srcImpl = (AbstractMemorySegmentImpl)srcSegment;
AbstractMemorySegmentImpl dstImpl = (AbstractMemorySegmentImpl)dstSegment;
if (srcElementLayout.byteSize() != dstElementLayout.byteSize()) {
- throw new IllegalArgumentException("Source and destination layouts must have same sizes");
+ throw new IllegalArgumentException("Source and destination layouts must have same size");
}
- if (srcOffset % srcElementLayout.byteAlignment() != 0) {
+ Utils.checkElementAlignment(srcElementLayout, "Source layout alignment greater than its size");
+ Utils.checkElementAlignment(dstElementLayout, "Destination layout alignment greater than its size");
+ if (!srcImpl.isAlignedForElement(srcOffset, srcElementLayout)) {
throw new IllegalArgumentException("Source segment incompatible with alignment constraints");
}
- if (dstOffset % dstElementLayout.byteAlignment() != 0) {
- throw new IllegalArgumentException("Target segment incompatible with alignment constraints");
+ if (!dstImpl.isAlignedForElement(dstOffset, dstElementLayout)) {
+ throw new IllegalArgumentException("Destination segment incompatible with alignment constraints");
}
long size = elementCount * srcElementLayout.byteSize();
srcImpl.checkAccess(srcOffset, size, true);
@@ -1034,6 +1090,8 @@ public sealed interface MemorySegment extends Addressable permits AbstractMemory
* @return a byte value read from this address.
* @throws IllegalStateException if the scope associated with this segment has been closed, or if access occurs from
* a thread other than the thread owning that scope.
+ * @throws IllegalArgumentException if the dereference operation is
+ * incompatible with the alignment constraints in the provided layout.
* @throws IndexOutOfBoundsException when the dereference operation falls outside the spatial bounds of the
* memory segment.
*/
@@ -1051,6 +1109,8 @@ public sealed interface MemorySegment extends Addressable permits AbstractMemory
* @param value the byte value to be written.
* @throws IllegalStateException if the scope associated with this segment has been closed, or if access occurs from
* a thread other than the thread owning that scope.
+ * @throws IllegalArgumentException if the dereference operation is
+ * incompatible with the alignment constraints in the provided layout.
* @throws IndexOutOfBoundsException when the dereference operation falls outside the spatial bounds of the
* memory segment.
* @throws UnsupportedOperationException if this segment is {@linkplain #isReadOnly() read-only}.
@@ -1069,6 +1129,8 @@ public sealed interface MemorySegment extends Addressable permits AbstractMemory
* @return a boolean value read from this address.
* @throws IllegalStateException if the scope associated with this segment has been closed, or if access occurs from
* a thread other than the thread owning that scope.
+ * @throws IllegalArgumentException if the dereference operation is
+ * incompatible with the alignment constraints in the provided layout.
* @throws IndexOutOfBoundsException when the dereference operation falls outside the spatial bounds of the
* memory segment.
*/
@@ -1086,6 +1148,8 @@ public sealed interface MemorySegment extends Addressable permits AbstractMemory
* @param value the boolean value to be written.
* @throws IllegalStateException if the scope associated with this segment has been closed, or if access occurs from
* a thread other than the thread owning that scope.
+ * @throws IllegalArgumentException if the dereference operation is
+ * incompatible with the alignment constraints in the provided layout.
* @throws IndexOutOfBoundsException when the dereference operation falls outside the spatial bounds of the
* memory segment.
* @throws UnsupportedOperationException if this segment is {@linkplain #isReadOnly() read-only}.
@@ -1104,6 +1168,8 @@ public sealed interface MemorySegment extends Addressable permits AbstractMemory
* @return a char value read from this address.
* @throws IllegalStateException if the scope associated with this segment has been closed, or if access occurs from
* a thread other than the thread owning that scope.
+ * @throws IllegalArgumentException if the dereference operation is
+ * incompatible with the alignment constraints in the provided layout.
* @throws IndexOutOfBoundsException when the dereference operation falls outside the spatial bounds of the
* memory segment.
*/
@@ -1121,6 +1187,8 @@ public sealed interface MemorySegment extends Addressable permits AbstractMemory
* @param value the char value to be written.
* @throws IllegalStateException if the scope associated with this segment has been closed, or if access occurs from
* a thread other than the thread owning that scope.
+ * @throws IllegalArgumentException if the dereference operation is
+ * incompatible with the alignment constraints in the provided layout.
* @throws IndexOutOfBoundsException when the dereference operation falls outside the spatial bounds of the
* memory segment.
* @throws UnsupportedOperationException if this segment is {@linkplain #isReadOnly() read-only}.
@@ -1139,6 +1207,8 @@ public sealed interface MemorySegment extends Addressable permits AbstractMemory
* @return a short value read from this address.
* @throws IllegalStateException if the scope associated with this segment has been closed, or if access occurs from
* a thread other than the thread owning that scope.
+ * @throws IllegalArgumentException if the dereference operation is
+ * incompatible with the alignment constraints in the provided layout.
* @throws IndexOutOfBoundsException when the dereference operation falls outside the spatial bounds of the
* memory segment.
*/
@@ -1156,6 +1226,8 @@ public sealed interface MemorySegment extends Addressable permits AbstractMemory
* @param value the short value to be written.
* @throws IllegalStateException if the scope associated with this segment has been closed, or if access occurs from
* a thread other than the thread owning that scope.
+ * @throws IllegalArgumentException if the dereference operation is
+ * incompatible with the alignment constraints in the provided layout.
* @throws IndexOutOfBoundsException when the dereference operation falls outside the spatial bounds of the
* memory segment.
* @throws UnsupportedOperationException if this segment is {@linkplain #isReadOnly() read-only}.
@@ -1174,6 +1246,8 @@ public sealed interface MemorySegment extends Addressable permits AbstractMemory
* @return an int value read from this address.
* @throws IllegalStateException if the scope associated with this segment has been closed, or if access occurs from
* a thread other than the thread owning that scope.
+ * @throws IllegalArgumentException if the dereference operation is
+ * incompatible with the alignment constraints in the provided layout.
* @throws IndexOutOfBoundsException when the dereference operation falls outside the spatial bounds of the
* memory segment.
*/
@@ -1191,6 +1265,8 @@ public sealed interface MemorySegment extends Addressable permits AbstractMemory
* @param value the int value to be written.
* @throws IllegalStateException if the scope associated with this segment has been closed, or if access occurs from
* a thread other than the thread owning that scope.
+ * @throws IllegalArgumentException if the dereference operation is
+ * incompatible with the alignment constraints in the provided layout.
* @throws IndexOutOfBoundsException when the dereference operation falls outside the spatial bounds of the
* memory segment.
* @throws UnsupportedOperationException if this segment is {@linkplain #isReadOnly() read-only}.
@@ -1209,6 +1285,8 @@ public sealed interface MemorySegment extends Addressable permits AbstractMemory
* @return a float value read from this address.
* @throws IllegalStateException if the scope associated with this segment has been closed, or if access occurs from
* a thread other than the thread owning that scope.
+ * @throws IllegalArgumentException if the dereference operation is
+ * incompatible with the alignment constraints in the provided layout.
* @throws IndexOutOfBoundsException when the dereference operation falls outside the spatial bounds of the
* memory segment.
*/
@@ -1226,6 +1304,8 @@ public sealed interface MemorySegment extends Addressable permits AbstractMemory
* @param value the float value to be written.
* @throws IllegalStateException if the scope associated with this segment has been closed, or if access occurs from
* a thread other than the thread owning that scope.
+ * @throws IllegalArgumentException if the dereference operation is
+ * incompatible with the alignment constraints in the provided layout.
* @throws IndexOutOfBoundsException when the dereference operation falls outside the spatial bounds of the
* memory segment.
* @throws UnsupportedOperationException if this segment is {@linkplain #isReadOnly() read-only}.
@@ -1244,6 +1324,8 @@ public sealed interface MemorySegment extends Addressable permits AbstractMemory
* @return a long value read from this address.
* @throws IllegalStateException if the scope associated with this segment has been closed, or if access occurs from
* a thread other than the thread owning that scope.
+ * @throws IllegalArgumentException if the dereference operation is
+ * incompatible with the alignment constraints in the provided layout.
* @throws IndexOutOfBoundsException when the dereference operation falls outside the spatial bounds of the
* memory segment.
*/
@@ -1261,6 +1343,8 @@ public sealed interface MemorySegment extends Addressable permits AbstractMemory
* @param value the long value to be written.
* @throws IllegalStateException if the scope associated with this segment has been closed, or if access occurs from
* a thread other than the thread owning that scope.
+ * @throws IllegalArgumentException if the dereference operation is
+ * incompatible with the alignment constraints in the provided layout.
* @throws IndexOutOfBoundsException when the dereference operation falls outside the spatial bounds of the
* memory segment.
* @throws UnsupportedOperationException if this segment is {@linkplain #isReadOnly() read-only}.
@@ -1279,6 +1363,8 @@ public sealed interface MemorySegment extends Addressable permits AbstractMemory
* @return a double value read from this address.
* @throws IllegalStateException if the scope associated with this segment has been closed, or if access occurs from
* a thread other than the thread owning that scope.
+ * @throws IllegalArgumentException if the dereference operation is
+ * incompatible with the alignment constraints in the provided layout.
* @throws IndexOutOfBoundsException when the dereference operation falls outside the spatial bounds of the
* memory segment.
*/
@@ -1296,6 +1382,8 @@ public sealed interface MemorySegment extends Addressable permits AbstractMemory
* @param value the double value to be written.
* @throws IllegalStateException if the scope associated with this segment has been closed, or if access occurs from
* a thread other than the thread owning that scope.
+ * @throws IllegalArgumentException if the dereference operation is
+ * incompatible with the alignment constraints in the provided layout.
* @throws IndexOutOfBoundsException when the dereference operation falls outside the spatial bounds of the
* memory segment.
* @throws UnsupportedOperationException if this segment is {@linkplain #isReadOnly() read-only}.
@@ -1314,6 +1402,8 @@ public sealed interface MemorySegment extends Addressable permits AbstractMemory
* @return an address value read from this address.
* @throws IllegalStateException if the scope associated with this segment has been closed, or if access occurs from
* a thread other than the thread owning that scope.
+ * @throws IllegalArgumentException if the dereference operation is
+ * incompatible with the alignment constraints in the provided layout.
* @throws IndexOutOfBoundsException when the dereference operation falls outside the spatial bounds of the
* memory segment.
*/
@@ -1331,6 +1421,8 @@ public sealed interface MemorySegment extends Addressable permits AbstractMemory
* @param value the address value to be written.
* @throws IllegalStateException if the scope associated with this segment has been closed, or if access occurs from
* a thread other than the thread owning that scope.
+ * @throws IllegalArgumentException if the dereference operation is
+ * incompatible with the alignment constraints in the provided layout.
* @throws IndexOutOfBoundsException when the dereference operation falls outside the spatial bounds of the
* memory segment.
* @throws UnsupportedOperationException if this segment is {@linkplain #isReadOnly() read-only}.
@@ -1349,11 +1441,15 @@ public sealed interface MemorySegment extends Addressable permits AbstractMemory
* @return a char value read from this address.
* @throws IllegalStateException if the scope associated with this segment has been closed, or if access occurs from
* a thread other than the thread owning that scope.
+ * @throws IllegalArgumentException if the dereference operation is
+ * incompatible with the alignment constraints in the provided layout,
+ * or if the layout alignment is greater than its size.
* @throws IndexOutOfBoundsException when the dereference operation falls outside the spatial bounds of the
* memory segment.
*/
@ForceInline
default char getAtIndex(ValueLayout.OfChar layout, long index) {
+ Utils.checkElementAlignment(layout, "Layout alignment greater than its size");
return (char)layout.accessHandle().get(this, Utils.scaleOffset(this, index, layout.byteSize()));
}
@@ -1366,12 +1462,16 @@ public sealed interface MemorySegment extends Addressable permits AbstractMemory
* @param value the char value to be written.
* @throws IllegalStateException if the scope associated with this segment has been closed, or if access occurs from
* a thread other than the thread owning that scope.
+ * @throws IllegalArgumentException if the dereference operation is
+ * incompatible with the alignment constraints in the provided layout,
+ * or if the layout alignment is greater than its size.
* @throws IndexOutOfBoundsException when the dereference operation falls outside the spatial bounds of the
* memory segment.
* @throws UnsupportedOperationException if this segment is {@linkplain #isReadOnly() read-only}.
*/
@ForceInline
default void setAtIndex(ValueLayout.OfChar layout, long index, char value) {
+ Utils.checkElementAlignment(layout, "Layout alignment greater than its size");
layout.accessHandle().set(this, Utils.scaleOffset(this, index, layout.byteSize()), value);
}
@@ -1384,11 +1484,15 @@ public sealed interface MemorySegment extends Addressable permits AbstractMemory
* @return a short value read from this address.
* @throws IllegalStateException if the scope associated with this segment has been closed, or if access occurs from
* a thread other than the thread owning that scope.
+ * @throws IllegalArgumentException if the dereference operation is
+ * incompatible with the alignment constraints in the provided layout,
+ * or if the layout alignment is greater than its size.
* @throws IndexOutOfBoundsException when the dereference operation falls outside the spatial bounds of the
* memory segment.
*/
@ForceInline
default short getAtIndex(ValueLayout.OfShort layout, long index) {
+ Utils.checkElementAlignment(layout, "Layout alignment greater than its size");
return (short)layout.accessHandle().get(this, Utils.scaleOffset(this, index, layout.byteSize()));
}
@@ -1401,12 +1505,16 @@ public sealed interface MemorySegment extends Addressable permits AbstractMemory
* @param value the short value to be written.
* @throws IllegalStateException if the scope associated with this segment has been closed, or if access occurs from
* a thread other than the thread owning that scope.
+ * @throws IllegalArgumentException if the dereference operation is
+ * incompatible with the alignment constraints in the provided layout,
+ * or if the layout alignment is greater than its size.
* @throws IndexOutOfBoundsException when the dereference operation falls outside the spatial bounds of the
* memory segment.
* @throws UnsupportedOperationException if this segment is {@linkplain #isReadOnly() read-only}.
*/
@ForceInline
default void setAtIndex(ValueLayout.OfShort layout, long index, short value) {
+ Utils.checkElementAlignment(layout, "Layout alignment greater than its size");
layout.accessHandle().set(this, Utils.scaleOffset(this, index, layout.byteSize()), value);
}
@@ -1419,11 +1527,15 @@ public sealed interface MemorySegment extends Addressable permits AbstractMemory
* @return an int value read from this address.
* @throws IllegalStateException if the scope associated with this segment has been closed, or if access occurs from
* a thread other than the thread owning that scope.
+ * @throws IllegalArgumentException if the dereference operation is
+ * incompatible with the alignment constraints in the provided layout,
+ * or if the layout alignment is greater than its size.
* @throws IndexOutOfBoundsException when the dereference operation falls outside the spatial bounds of the
* memory segment.
*/
@ForceInline
default int getAtIndex(ValueLayout.OfInt layout, long index) {
+ Utils.checkElementAlignment(layout, "Layout alignment greater than its size");
return (int)layout.accessHandle().get(this, Utils.scaleOffset(this, index, layout.byteSize()));
}
@@ -1436,12 +1548,16 @@ public sealed interface MemorySegment extends Addressable permits AbstractMemory
* @param value the int value to be written.
* @throws IllegalStateException if the scope associated with this segment has been closed, or if access occurs from
* a thread other than the thread owning that scope.
+ * @throws IllegalArgumentException if the dereference operation is
+ * incompatible with the alignment constraints in the provided layout,
+ * or if the layout alignment is greater than its size.
* @throws IndexOutOfBoundsException when the dereference operation falls outside the spatial bounds of the
* memory segment.
* @throws UnsupportedOperationException if this segment is {@linkplain #isReadOnly() read-only}.
*/
@ForceInline
default void setAtIndex(ValueLayout.OfInt layout, long index, int value) {
+ Utils.checkElementAlignment(layout, "Layout alignment greater than its size");
layout.accessHandle().set(this, Utils.scaleOffset(this, index, layout.byteSize()), value);
}
@@ -1454,11 +1570,15 @@ public sealed interface MemorySegment extends Addressable permits AbstractMemory
* @return a float value read from this address.
* @throws IllegalStateException if the scope associated with this segment has been closed, or if access occurs from
* a thread other than the thread owning that scope.
+ * @throws IllegalArgumentException if the dereference operation is
+ * incompatible with the alignment constraints in the provided layout,
+ * or if the layout alignment is greater than its size.
* @throws IndexOutOfBoundsException when the dereference operation falls outside the spatial bounds of the
* memory segment.
*/
@ForceInline
default float getAtIndex(ValueLayout.OfFloat layout, long index) {
+ Utils.checkElementAlignment(layout, "Layout alignment greater than its size");
return (float)layout.accessHandle().get(this, Utils.scaleOffset(this, index, layout.byteSize()));
}
@@ -1471,12 +1591,16 @@ public sealed interface MemorySegment extends Addressable permits AbstractMemory
* @param value the float value to be written.
* @throws IllegalStateException if the scope associated with this segment has been closed, or if access occurs from
* a thread other than the thread owning that scope.
+ * @throws IllegalArgumentException if the dereference operation is
+ * incompatible with the alignment constraints in the provided layout,
+ * or if the layout alignment is greater than its size.
* @throws IndexOutOfBoundsException when the dereference operation falls outside the spatial bounds of the
* memory segment.
* @throws UnsupportedOperationException if this segment is {@linkplain #isReadOnly() read-only}.
*/
@ForceInline
default void setAtIndex(ValueLayout.OfFloat layout, long index, float value) {
+ Utils.checkElementAlignment(layout, "Layout alignment greater than its size");
layout.accessHandle().set(this, Utils.scaleOffset(this, index, layout.byteSize()), value);
}
@@ -1489,11 +1613,15 @@ public sealed interface MemorySegment extends Addressable permits AbstractMemory
* @return a long value read from this address.
* @throws IllegalStateException if the scope associated with this segment has been closed, or if access occurs from
* a thread other than the thread owning that scope.
+ * @throws IllegalArgumentException if the dereference operation is
+ * incompatible with the alignment constraints in the provided layout,
+ * or if the layout alignment is greater than its size.
* @throws IndexOutOfBoundsException when the dereference operation falls outside the spatial bounds of the
* memory segment.
*/
@ForceInline
default long getAtIndex(ValueLayout.OfLong layout, long index) {
+ Utils.checkElementAlignment(layout, "Layout alignment greater than its size");
return (long)layout.accessHandle().get(this, Utils.scaleOffset(this, index, layout.byteSize()));
}
@@ -1506,12 +1634,16 @@ public sealed interface MemorySegment extends Addressable permits AbstractMemory
* @param value the long value to be written.
* @throws IllegalStateException if the scope associated with this segment has been closed, or if access occurs from
* a thread other than the thread owning that scope.
+ * @throws IllegalArgumentException if the dereference operation is
+ * incompatible with the alignment constraints in the provided layout,
+ * or if the layout alignment is greater than its size.
* @throws IndexOutOfBoundsException when the dereference operation falls outside the spatial bounds of the
* memory segment.
* @throws UnsupportedOperationException if this segment is {@linkplain #isReadOnly() read-only}.
*/
@ForceInline
default void setAtIndex(ValueLayout.OfLong layout, long index, long value) {
+ Utils.checkElementAlignment(layout, "Layout alignment greater than its size");
layout.accessHandle().set(this, Utils.scaleOffset(this, index, layout.byteSize()), value);
}
@@ -1524,11 +1656,15 @@ public sealed interface MemorySegment extends Addressable permits AbstractMemory
* @return a double value read from this address.
* @throws IllegalStateException if the scope associated with this segment has been closed, or if access occurs from
* a thread other than the thread owning that scope.
+ * @throws IllegalArgumentException if the dereference operation is
+ * incompatible with the alignment constraints in the provided layout,
+ * or if the layout alignment is greater than its size.
* @throws IndexOutOfBoundsException when the dereference operation falls outside the spatial bounds of the
* memory segment.
*/
@ForceInline
default double getAtIndex(ValueLayout.OfDouble layout, long index) {
+ Utils.checkElementAlignment(layout, "Layout alignment greater than its size");
return (double)layout.accessHandle().get(this, Utils.scaleOffset(this, index, layout.byteSize()));
}
@@ -1541,12 +1677,16 @@ public sealed interface MemorySegment extends Addressable permits AbstractMemory
* @param value the double value to be written.
* @throws IllegalStateException if the scope associated with this segment has been closed, or if access occurs from
* a thread other than the thread owning that scope.
+ * @throws IllegalArgumentException if the dereference operation is
+ * incompatible with the alignment constraints in the provided layout,
+ * or if the layout alignment is greater than its size.
* @throws IndexOutOfBoundsException when the dereference operation falls outside the spatial bounds of the
* memory segment.
* @throws UnsupportedOperationException if this segment is {@linkplain #isReadOnly() read-only}.
*/
@ForceInline
default void setAtIndex(ValueLayout.OfDouble layout, long index, double value) {
+ Utils.checkElementAlignment(layout, "Layout alignment greater than its size");
layout.accessHandle().set(this, Utils.scaleOffset(this, index, layout.byteSize()), value);
}
@@ -1559,11 +1699,15 @@ public sealed interface MemorySegment extends Addressable permits AbstractMemory
* @return an address value read from this address.
* @throws IllegalStateException if the scope associated with this segment has been closed, or if access occurs from
* a thread other than the thread owning that scope.
+ * @throws IllegalArgumentException if the dereference operation is
+ * incompatible with the alignment constraints in the provided layout,
+ * or if the layout alignment is greater than its size.
* @throws IndexOutOfBoundsException when the dereference operation falls outside the spatial bounds of the
* memory segment.
*/
@ForceInline
default MemoryAddress getAtIndex(ValueLayout.OfAddress layout, long index) {
+ Utils.checkElementAlignment(layout, "Layout alignment greater than its size");
return (MemoryAddress)layout.accessHandle().get(this, Utils.scaleOffset(this, index, layout.byteSize()));
}
@@ -1576,12 +1720,16 @@ public sealed interface MemorySegment extends Addressable permits AbstractMemory
* @param value the address value to be written.
* @throws IllegalStateException if the scope associated with this segment has been closed, or if access occurs from
* a thread other than the thread owning that scope.
+ * @throws IllegalArgumentException if the dereference operation is
+ * incompatible with the alignment constraints in the provided layout,
+ * or if the layout alignment is greater than its size.
* @throws IndexOutOfBoundsException when the dereference operation falls outside the spatial bounds of the
* memory segment.
* @throws UnsupportedOperationException if this segment is {@linkplain #isReadOnly() read-only}.
*/
@ForceInline
default void setAtIndex(ValueLayout.OfAddress layout, long index, Addressable value) {
+ Utils.checkElementAlignment(layout, "Layout alignment greater than its size");
layout.accessHandle().set(this, Utils.scaleOffset(this, index, layout.byteSize()), value.address());
}
@@ -1598,7 +1746,9 @@ public sealed interface MemorySegment extends Addressable permits AbstractMemory
* @param dstIndex the starting index of the destination array.
* @param elementCount the number of array elements to be copied.
* @throws IllegalArgumentException if {@code dstArray} is not an array, or if it is an array but whose type is not supported,
- * or if the destination array component type does not match the carrier of the source element layout.
+ * if the destination array component type does not match the carrier of the source element layout, if the source
+ * segment/offset are incompatible with the alignment constraints in the source element layout,
+ * or if the destination element layout alignment is greater than its size.
*/
@ForceInline
static void copy(
@@ -1614,6 +1764,10 @@ public sealed interface MemorySegment extends Addressable permits AbstractMemory
int dstBase = (int)baseAndScale;
int dstWidth = (int)(baseAndScale >> 32);
AbstractMemorySegmentImpl srcImpl = (AbstractMemorySegmentImpl)srcSegment;
+ Utils.checkElementAlignment(srcLayout, "Source layout alignment greater than its size");
+ if (!srcImpl.isAlignedForElement(srcOffset, srcLayout)) {
+ throw new IllegalArgumentException("Source segment incompatible with alignment constraints");
+ }
srcImpl.checkAccess(srcOffset, elementCount * dstWidth, true);
Objects.checkFromIndexSize(dstIndex, elementCount, Array.getLength(dstArray));
if (dstWidth == 1 || srcLayout.order() == ByteOrder.nativeOrder()) {
@@ -1639,7 +1793,9 @@ public sealed interface MemorySegment extends Addressable permits AbstractMemory
* @param dstOffset the starting offset, in bytes, of the destination segment.
* @param elementCount the number of array elements to be copied.
* @throws IllegalArgumentException if {@code srcArray} is not an array, or if it is an array but whose type is not supported,
- * or if the source array component type does not match the carrier of the destination element layout.
+ * if the source array component type does not match the carrier of the destination element layout, if the destination
+ * segment/offset are incompatible with the alignment constraints in the destination element layout,
+ * or if the destination element layout alignment is greater than its size.
*/
@ForceInline
static void copy(
@@ -1656,6 +1812,10 @@ public sealed interface MemorySegment extends Addressable permits AbstractMemory
int srcWidth = (int)(baseAndScale >> 32);
Objects.checkFromIndexSize(srcIndex, elementCount, Array.getLength(srcArray));
AbstractMemorySegmentImpl destImpl = (AbstractMemorySegmentImpl)dstSegment;
+ Utils.checkElementAlignment(dstLayout, "Destination layout alignment greater than its size");
+ if (!destImpl.isAlignedForElement(dstOffset, dstLayout)) {
+ throw new IllegalArgumentException("Destination segment incompatible with alignment constraints");
+ }
destImpl.checkAccess(dstOffset, elementCount * srcWidth, false);
if (srcWidth == 1 || dstLayout.order() == ByteOrder.nativeOrder()) {
ScopedMemoryAccess.getScopedMemoryAccess().copyMemory(null, destImpl.scope(),
diff --git a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/AbstractMemorySegmentImpl.java b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/AbstractMemorySegmentImpl.java
index eacbeb26a51..5744d8556b1 100644
--- a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/AbstractMemorySegmentImpl.java
+++ b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/AbstractMemorySegmentImpl.java
@@ -124,8 +124,12 @@ public abstract non-sealed class AbstractMemorySegmentImpl extends MemorySegment
if (elementLayout.byteSize() == 0) {
throw new IllegalArgumentException("Element layout size cannot be zero");
}
- if (byteSize() % elementLayout.byteSize() != 0) {
- throw new IllegalArgumentException("Segment size is no a multiple of layout size");
+ Utils.checkElementAlignment(elementLayout, "Element layout alignment greater than its size");
+ if (!isAlignedForElement(0, elementLayout)) {
+ throw new IllegalArgumentException("Incompatible alignment constraints");
+ }
+ if (!Utils.isAligned(byteSize(), elementLayout.byteSize())) {
+ throw new IllegalArgumentException("Segment size is not a multiple of layout size");
}
return new SegmentSplitter(elementLayout.byteSize(), byteSize() / elementLayout.byteSize(),
this);
@@ -383,8 +387,13 @@ public abstract non-sealed class AbstractMemorySegmentImpl extends MemorySegment
return (this.mask & mask) != 0;
}
+ @ForceInline
+ public final boolean isAlignedForElement(long offset, MemoryLayout layout) {
+ return (((unsafeGetOffset() + offset) | maxAlignMask()) & (layout.byteAlignment() - 1)) == 0;
+ }
+
private int checkArraySize(String typeName, int elemSize) {
- if (length % elemSize != 0) {
+ if (!Utils.isAligned(length, elemSize)) {
throw new IllegalStateException(String.format("Segment size is not a multiple of %d. Size: %d", elemSize, length));
}
long arraySize = length / elemSize;
diff --git a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/HeapMemorySegmentImpl.java b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/HeapMemorySegmentImpl.java
index 755d6cb14dc..b10b278c0ca 100644
--- a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/HeapMemorySegmentImpl.java
+++ b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/HeapMemorySegmentImpl.java
@@ -47,6 +47,11 @@ public abstract class HeapMemorySegmentImpl extends AbstractMemorySegmentImpl
private static final Unsafe UNSAFE = Unsafe.getUnsafe();
private static final int BYTE_ARR_BASE = UNSAFE.arrayBaseOffset(byte[].class);
+ private static final long MAX_ALIGN_1 = 1;
+ private static final long MAX_ALIGN_2 = 2;
+ private static final long MAX_ALIGN_4 = 4;
+ private static final long MAX_ALIGN_8 = 8;
+
final long offset;
final H base;
@@ -100,6 +105,11 @@ public abstract class HeapMemorySegmentImpl extends AbstractMemorySegmentImpl
long byteSize = (long)arr.length * Unsafe.ARRAY_BYTE_INDEX_SCALE;
return new OfByte(Unsafe.ARRAY_BYTE_BASE_OFFSET, arr, byteSize, defaultAccessModes(byteSize));
}
+
+ @Override
+ public long maxAlignMask() {
+ return MAX_ALIGN_1;
+ }
}
public static class OfChar extends HeapMemorySegmentImpl {
@@ -123,6 +133,11 @@ public abstract class HeapMemorySegmentImpl extends AbstractMemorySegmentImpl
long byteSize = (long)arr.length * Unsafe.ARRAY_CHAR_INDEX_SCALE;
return new OfChar(Unsafe.ARRAY_CHAR_BASE_OFFSET, arr, byteSize, defaultAccessModes(byteSize));
}
+
+ @Override
+ public long maxAlignMask() {
+ return MAX_ALIGN_2;
+ }
}
public static class OfShort extends HeapMemorySegmentImpl {
@@ -146,6 +161,11 @@ public abstract class HeapMemorySegmentImpl extends AbstractMemorySegmentImpl
long byteSize = (long)arr.length * Unsafe.ARRAY_SHORT_INDEX_SCALE;
return new OfShort(Unsafe.ARRAY_SHORT_BASE_OFFSET, arr, byteSize, defaultAccessModes(byteSize));
}
+
+ @Override
+ public long maxAlignMask() {
+ return MAX_ALIGN_2;
+ }
}
public static class OfInt extends HeapMemorySegmentImpl {
@@ -169,6 +189,11 @@ public abstract class HeapMemorySegmentImpl extends AbstractMemorySegmentImpl
long byteSize = (long)arr.length * Unsafe.ARRAY_INT_INDEX_SCALE;
return new OfInt(Unsafe.ARRAY_INT_BASE_OFFSET, arr, byteSize, defaultAccessModes(byteSize));
}
+
+ @Override
+ public long maxAlignMask() {
+ return MAX_ALIGN_4;
+ }
}
public static class OfLong extends HeapMemorySegmentImpl {
@@ -192,6 +217,11 @@ public abstract class HeapMemorySegmentImpl extends AbstractMemorySegmentImpl
long byteSize = (long)arr.length * Unsafe.ARRAY_LONG_INDEX_SCALE;
return new OfLong(Unsafe.ARRAY_LONG_BASE_OFFSET, arr, byteSize, defaultAccessModes(byteSize));
}
+
+ @Override
+ public long maxAlignMask() {
+ return MAX_ALIGN_8;
+ }
}
public static class OfFloat extends HeapMemorySegmentImpl {
@@ -215,6 +245,11 @@ public abstract class HeapMemorySegmentImpl extends AbstractMemorySegmentImpl
long byteSize = (long)arr.length * Unsafe.ARRAY_FLOAT_INDEX_SCALE;
return new OfFloat(Unsafe.ARRAY_FLOAT_BASE_OFFSET, arr, byteSize, defaultAccessModes(byteSize));
}
+
+ @Override
+ public long maxAlignMask() {
+ return MAX_ALIGN_4;
+ }
}
public static class OfDouble extends HeapMemorySegmentImpl {
@@ -238,6 +273,11 @@ public abstract class HeapMemorySegmentImpl extends AbstractMemorySegmentImpl
long byteSize = (long)arr.length * Unsafe.ARRAY_DOUBLE_INDEX_SCALE;
return new OfDouble(Unsafe.ARRAY_DOUBLE_BASE_OFFSET, arr, byteSize, defaultAccessModes(byteSize));
}
+
+ @Override
+ public long maxAlignMask() {
+ return MAX_ALIGN_8;
+ }
}
}
diff --git a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/LayoutPath.java b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/LayoutPath.java
index 1d8bb265c83..dbe52a69799 100644
--- a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/LayoutPath.java
+++ b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/LayoutPath.java
@@ -282,11 +282,11 @@ public class LayoutPath {
private static void checkAlignment(LayoutPath path) {
MemoryLayout layout = path.layout;
long alignment = layout.bitAlignment();
- if (path.offset % alignment != 0) {
+ if (!Utils.isAligned(path.offset, alignment)) {
throw new UnsupportedOperationException("Invalid alignment requirements for layout " + layout);
}
for (long stride : path.strides) {
- if (stride % alignment != 0) {
+ if (!Utils.isAligned(stride, alignment)) {
throw new UnsupportedOperationException("Alignment requirements for layout " + layout + " do not match stride " + stride);
}
}
diff --git a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/MemoryAddressImpl.java b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/MemoryAddressImpl.java
index 59d7105082c..7bb55907571 100644
--- a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/MemoryAddressImpl.java
+++ b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/MemoryAddressImpl.java
@@ -267,6 +267,7 @@ public final class MemoryAddressImpl implements MemoryAddress, Scoped {
@CallerSensitive
public char getAtIndex(ValueLayout.OfChar layout, long index) {
Reflection.ensureNativeAccess(Reflection.getCallerClass());
+ Utils.checkElementAlignment(layout, "Layout alignment greater than its size");
return NativeMemorySegmentImpl.EVERYTHING.get(layout, toRawLongValue() + (index * layout.byteSize()));
}
@@ -275,6 +276,7 @@ public final class MemoryAddressImpl implements MemoryAddress, Scoped {
@CallerSensitive
public void setAtIndex(ValueLayout.OfChar layout, long index, char value) {
Reflection.ensureNativeAccess(Reflection.getCallerClass());
+ Utils.checkElementAlignment(layout, "Layout alignment greater than its size");
NativeMemorySegmentImpl.EVERYTHING.set(layout, toRawLongValue() + (index * layout.byteSize()), value);
}
@@ -283,6 +285,7 @@ public final class MemoryAddressImpl implements MemoryAddress, Scoped {
@CallerSensitive
public short getAtIndex(ValueLayout.OfShort layout, long index) {
Reflection.ensureNativeAccess(Reflection.getCallerClass());
+ Utils.checkElementAlignment(layout, "Layout alignment greater than its size");
return NativeMemorySegmentImpl.EVERYTHING.get(layout, toRawLongValue() + (index * layout.byteSize()));
}
@@ -291,6 +294,7 @@ public final class MemoryAddressImpl implements MemoryAddress, Scoped {
@CallerSensitive
public void setAtIndex(ValueLayout.OfShort layout, long index, short value) {
Reflection.ensureNativeAccess(Reflection.getCallerClass());
+ Utils.checkElementAlignment(layout, "Layout alignment greater than its size");
NativeMemorySegmentImpl.EVERYTHING.set(layout, toRawLongValue() + (index * layout.byteSize()), value);
}
@@ -299,6 +303,7 @@ public final class MemoryAddressImpl implements MemoryAddress, Scoped {
@CallerSensitive
public int getAtIndex(ValueLayout.OfInt layout, long index) {
Reflection.ensureNativeAccess(Reflection.getCallerClass());
+ Utils.checkElementAlignment(layout, "Layout alignment greater than its size");
return NativeMemorySegmentImpl.EVERYTHING.get(layout, toRawLongValue() + (index * layout.byteSize()));
}
@@ -307,6 +312,7 @@ public final class MemoryAddressImpl implements MemoryAddress, Scoped {
@CallerSensitive
public void setAtIndex(ValueLayout.OfInt layout, long index, int value) {
Reflection.ensureNativeAccess(Reflection.getCallerClass());
+ Utils.checkElementAlignment(layout, "Layout alignment greater than its size");
NativeMemorySegmentImpl.EVERYTHING.set(layout, toRawLongValue() + (index * layout.byteSize()), value);
}
@@ -315,6 +321,7 @@ public final class MemoryAddressImpl implements MemoryAddress, Scoped {
@CallerSensitive
public float getAtIndex(ValueLayout.OfFloat layout, long index) {
Reflection.ensureNativeAccess(Reflection.getCallerClass());
+ Utils.checkElementAlignment(layout, "Layout alignment greater than its size");
return NativeMemorySegmentImpl.EVERYTHING.get(layout, toRawLongValue() + (index * layout.byteSize()));
}
@@ -323,6 +330,7 @@ public final class MemoryAddressImpl implements MemoryAddress, Scoped {
@CallerSensitive
public void setAtIndex(ValueLayout.OfFloat layout, long index, float value) {
Reflection.ensureNativeAccess(Reflection.getCallerClass());
+ Utils.checkElementAlignment(layout, "Layout alignment greater than its size");
NativeMemorySegmentImpl.EVERYTHING.set(layout, toRawLongValue() + (index * layout.byteSize()), value);
}
@@ -331,6 +339,7 @@ public final class MemoryAddressImpl implements MemoryAddress, Scoped {
@CallerSensitive
public long getAtIndex(ValueLayout.OfLong layout, long index) {
Reflection.ensureNativeAccess(Reflection.getCallerClass());
+ Utils.checkElementAlignment(layout, "Layout alignment greater than its size");
return NativeMemorySegmentImpl.EVERYTHING.get(layout, toRawLongValue() + (index * layout.byteSize()));
}
@@ -339,6 +348,7 @@ public final class MemoryAddressImpl implements MemoryAddress, Scoped {
@CallerSensitive
public void setAtIndex(ValueLayout.OfLong layout, long index, long value) {
Reflection.ensureNativeAccess(Reflection.getCallerClass());
+ Utils.checkElementAlignment(layout, "Layout alignment greater than its size");
NativeMemorySegmentImpl.EVERYTHING.set(layout, toRawLongValue() + (index * layout.byteSize()), value);
}
@@ -347,6 +357,7 @@ public final class MemoryAddressImpl implements MemoryAddress, Scoped {
@CallerSensitive
public double getAtIndex(ValueLayout.OfDouble layout, long index) {
Reflection.ensureNativeAccess(Reflection.getCallerClass());
+ Utils.checkElementAlignment(layout, "Layout alignment greater than its size");
return NativeMemorySegmentImpl.EVERYTHING.get(layout, toRawLongValue() + (index * layout.byteSize()));
}
@@ -355,6 +366,7 @@ public final class MemoryAddressImpl implements MemoryAddress, Scoped {
@CallerSensitive
public void setAtIndex(ValueLayout.OfDouble layout, long index, double value) {
Reflection.ensureNativeAccess(Reflection.getCallerClass());
+ Utils.checkElementAlignment(layout, "Layout alignment greater than its size");
NativeMemorySegmentImpl.EVERYTHING.set(layout, toRawLongValue() + (index * layout.byteSize()), value);
}
@@ -363,6 +375,7 @@ public final class MemoryAddressImpl implements MemoryAddress, Scoped {
@CallerSensitive
public MemoryAddress getAtIndex(ValueLayout.OfAddress layout, long index) {
Reflection.ensureNativeAccess(Reflection.getCallerClass());
+ Utils.checkElementAlignment(layout, "Layout alignment greater than its size");
return NativeMemorySegmentImpl.EVERYTHING.get(layout, toRawLongValue() + (index * layout.byteSize()));
}
@@ -371,6 +384,7 @@ public final class MemoryAddressImpl implements MemoryAddress, Scoped {
@CallerSensitive
public void setAtIndex(ValueLayout.OfAddress layout, long index, Addressable value) {
Reflection.ensureNativeAccess(Reflection.getCallerClass());
+ Utils.checkElementAlignment(layout, "Layout alignment greater than its size");
NativeMemorySegmentImpl.EVERYTHING.set(layout, toRawLongValue() + (index * layout.byteSize()), value.address());
}
}
diff --git a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/NativeMemorySegmentImpl.java b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/NativeMemorySegmentImpl.java
index 11f795a2f84..bd06223b6ce 100644
--- a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/NativeMemorySegmentImpl.java
+++ b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/NativeMemorySegmentImpl.java
@@ -92,6 +92,11 @@ public class NativeMemorySegmentImpl extends AbstractMemorySegmentImpl {
return null;
}
+ @Override
+ public long maxAlignMask() {
+ return 0;
+ }
+
// factories
public static MemorySegment makeNativeSegment(long bytesSize, long alignmentBytes, ResourceScopeImpl scope) {
diff --git a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/Utils.java b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/Utils.java
index 6272e6e3100..bb9db16eff2 100644
--- a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/Utils.java
+++ b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/Utils.java
@@ -98,7 +98,7 @@ public final class Utils {
}
public static long bitsToBytesOrThrow(long bits, Supplier exFactory) {
- if (bits % 8 == 0) {
+ if (Utils.isAligned(bits, 8)) {
return bits / 8;
} else {
throw exFactory.get();
@@ -173,4 +173,16 @@ public final class Utils {
// note: we know size is a small value (as it comes from ValueLayout::byteSize())
return MemorySegmentProxy.multiplyOffsets(index, (int)size, (AbstractMemorySegmentImpl)segment);
}
+
+ @ForceInline
+ public static boolean isAligned(long offset, long align) {
+ return (offset & (align - 1)) == 0;
+ }
+
+ @ForceInline
+ public static void checkElementAlignment(MemoryLayout layout, String msg) {
+ if (layout.byteAlignment() > layout.byteSize()) {
+ throw new IllegalArgumentException(msg);
+ }
+ }
}
diff --git a/test/hotspot/jtreg/compiler/vectorapi/Test8278948.java b/test/hotspot/jtreg/compiler/vectorapi/Test8278948.java
new file mode 100644
index 00000000000..5218e10c4af
--- /dev/null
+++ b/test/hotspot/jtreg/compiler/vectorapi/Test8278948.java
@@ -0,0 +1,82 @@
+/*
+ * Copyright (c) 2022, Oracle and/or its affiliates. All rights reserved.
+ * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
+ *
+ * This code is free software; you can redistribute it and/or modify it
+ * under the terms of the GNU General Public License version 2 only, as
+ * published by the Free Software Foundation.
+ *
+ * This code is distributed in the hope that it will be useful, but WITHOUT
+ * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
+ * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
+ * version 2 for more details (a copy is included in the LICENSE file that
+ * accompanied this code).
+ *
+ * You should have received a copy of the GNU General Public License version
+ * 2 along with this work; if not, write to the Free Software Foundation,
+ * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
+ *
+ * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
+ * or visit www.oracle.com if you need additional information or have any
+ * questions.
+ */
+
+package compiler.vectorapi;
+
+import java.util.Random;
+import jdk.incubator.vector.ByteVector;
+import jdk.incubator.vector.DoubleVector;
+import jdk.incubator.vector.ShortVector;
+import jdk.test.lib.Asserts;
+import jdk.test.lib.Utils;
+
+/*
+ * @test
+ * @bug 8278948
+ * @summary Intermediate integer promotion vector length encoding is calculated incorrectly on x86
+ * @modules jdk.incubator.vector
+ * @library /test/lib
+ * @run main/othervm -XX:+IgnoreUnrecognizedVMOptions -XX:-TieredCompilation -XX:CompileThreshold=100 -XX:UseAVX=1
+ * compiler.vectorapi.Test8278948
+ */
+public class Test8278948 {
+ static final int INVOCATIONS = 10000;
+
+ static final Random random = Utils.getRandomInstance();
+ static final byte[] BYTES = new byte[8];
+ static final short[] SHORTS = new short[4];
+ static final double[] DOUBLES = new double[4];
+
+
+ public static void main(String[] args) {
+ for (int i = 0; i < INVOCATIONS; i++) {
+ for (int j = 0; j < DOUBLES.length; j++) {
+ BYTES[j] = (byte)random.nextInt();
+ }
+ bytesToDoubles();
+ for (int j = 0; j < DOUBLES.length; j++) {
+ Asserts.assertEquals((double)BYTES[j], DOUBLES[j]);
+ }
+
+ for (int j = 0; j < DOUBLES.length; j++) {
+ SHORTS[j] = (short)random.nextInt();
+ }
+ shortsToDoubles();
+ for (int j = 0; j < DOUBLES.length; j++) {
+ Asserts.assertEquals((double)SHORTS[j], DOUBLES[j]);
+ }
+ }
+ }
+
+ static void bytesToDoubles() {
+ ((DoubleVector)ByteVector.fromArray(ByteVector.SPECIES_64, BYTES, 0)
+ .castShape(DoubleVector.SPECIES_256, 0))
+ .intoArray(DOUBLES, 0);
+ }
+
+ static void shortsToDoubles() {
+ ((DoubleVector)ShortVector.fromArray(ShortVector.SPECIES_64, SHORTS, 0)
+ .castShape(DoubleVector.SPECIES_256, 0))
+ .intoArray(DOUBLES, 0);
+ }
+}
diff --git a/test/jdk/ProblemList.txt b/test/jdk/ProblemList.txt
index 2bd5819f82d..33e66e317a3 100644
--- a/test/jdk/ProblemList.txt
+++ b/test/jdk/ProblemList.txt
@@ -517,6 +517,7 @@ java/awt/KeyboardFocusmanager/TypeAhead/ButtonActionKeyTest/ButtonActionKeyTest.
java/awt/Window/GetScreenLocation/GetScreenLocationTest.java 8225787 linux-x64
java/awt/Dialog/MakeWindowAlwaysOnTop/MakeWindowAlwaysOnTop.java 8266243 macosx-aarch64
java/awt/dnd/BadSerializationTest/BadSerializationTest.java 8277817 linux-x64,windows-x64
+java/awt/GraphicsDevice/CheckDisplayModes.java 8266242 macosx-aarch64
############################################################################
@@ -606,6 +607,8 @@ java/nio/channels/DatagramChannel/Unref.java 8233519 generic-
java/nio/channels/AsynchronousSocketChannel/StressLoopback.java 8211851 aix-ppc64
+java/nio/channels/DatagramChannel/ManySourcesAndTargets.java 8264385 macosx-aarch64
+
############################################################################
# jdk_rmi
@@ -669,6 +672,7 @@ javax/security/auth/kerberos/KerberosTixDateTest.java 8039280 generic-
sun/security/provider/PolicyFile/GrantAllPermToExtWhenNoPolicy.java 8039280 generic-all
sun/security/provider/PolicyParser/ExtDirsChange.java 8039280 generic-all
sun/security/provider/PolicyParser/PrincipalExpansionError.java 8039280 generic-all
+sun/security/ssl/SSLSessionImpl/NoInvalidateSocketException.java 8277970 linux-all,macosx-x64
############################################################################
diff --git a/test/jdk/java/foreign/TestArrayCopy.java b/test/jdk/java/foreign/TestArrayCopy.java
index b8a36307fde..e45fcf395f7 100644
--- a/test/jdk/java/foreign/TestArrayCopy.java
+++ b/test/jdk/java/foreign/TestArrayCopy.java
@@ -253,6 +253,18 @@ public class TestArrayCopy {
MemorySegment.copy(new byte[] { 1, 2, 3, 4 }, 0, segment, JAVA_INT, 0, 4);
}
+ @Test(expectedExceptions = IllegalArgumentException.class)
+ public void testHyperAlignedSrc() {
+ MemorySegment segment = MemorySegment.ofArray(new byte[] {1, 2, 3, 4});
+ MemorySegment.copy(new byte[] { 1, 2, 3, 4 }, 0, segment, JAVA_BYTE.withBitAlignment(16), 0, 4);
+ }
+
+ @Test(expectedExceptions = IllegalArgumentException.class)
+ public void testHyperAlignedDst() {
+ MemorySegment segment = MemorySegment.ofArray(new byte[] {1, 2, 3, 4});
+ MemorySegment.copy(segment, JAVA_BYTE.withBitAlignment(16), 0, new byte[] { 1, 2, 3, 4 }, 0, 4);
+ }
+
/***** Utilities *****/
public static MemorySegment srcSegment(int bytesLength) {
diff --git a/test/jdk/java/foreign/TestHeapAlignment.java b/test/jdk/java/foreign/TestHeapAlignment.java
new file mode 100644
index 00000000000..3395a686f38
--- /dev/null
+++ b/test/jdk/java/foreign/TestHeapAlignment.java
@@ -0,0 +1,131 @@
+/*
+ * Copyright (c) 2021, Oracle and/or its affiliates. All rights reserved.
+ * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
+ *
+ * This code is free software; you can redistribute it and/or modify it
+ * under the terms of the GNU General Public License version 2 only, as
+ * published by the Free Software Foundation.
+ *
+ * This code is distributed in the hope that it will be useful, but WITHOUT
+ * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
+ * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
+ * version 2 for more details (a copy is included in the LICENSE file that
+ * accompanied this code).
+ *
+ * You should have received a copy of the GNU General Public License version
+ * 2 along with this work; if not, write to the Free Software Foundation,
+ * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
+ *
+ * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
+ * or visit www.oracle.com if you need additional information or have any
+ * questions.
+ *
+ */
+
+/*
+ * @test
+ * @requires ((os.arch == "amd64" | os.arch == "x86_64") & sun.arch.data.model == "64") | os.arch == "aarch64"
+ * @run testng/othervm --enable-native-access=ALL-UNNAMED TestHeapAlignment
+ */
+
+import jdk.incubator.foreign.MemoryAddress;
+import jdk.incubator.foreign.MemoryLayout;
+import jdk.incubator.foreign.MemorySegment;
+import jdk.incubator.foreign.ResourceScope;
+import jdk.incubator.foreign.ValueLayout;
+import org.testng.annotations.DataProvider;
+import org.testng.annotations.Test;
+
+import java.util.ArrayList;
+import java.util.List;
+import java.util.function.Function;
+
+import static org.testng.Assert.fail;
+
+public class TestHeapAlignment {
+
+ @Test(dataProvider = "layouts")
+ public void testHeapAlignment(MemorySegment segment, int align, Object val, Object arr, ValueLayout layout, Function