Attribute safety
Adam Sotona
adam.sotona at oracle.com
Mon Jul 31 13:19:41 UTC 2023
I’ve added some factories below, which may simplify handling of some common custom attribute cases, for example:
Classfile.of(Classfile.AttributeMapperOption.of(attrName -> switch (attrName.stringValue()) {
case "MyCustomBinaryContentAttribute" -> selfContainedCustomAttribute(attrName.stringValue());
case "MyCustomAttributeWithCPReference" -> singleConstantPoolEntryCustomAttribute(attrName.stringValue());
default -> null;
}));
For category #1 there might be a single factory getting attribute name and returning attribute mapper.
static <T extends CustomAttribute<T>>
AttributeMapper<T> selfContainedCustomAttribute(String attributeName,
BiFunction<AttributeMapper<T>, byte[], T> attributeFactory,
Function<T, byte[]> contentAccessor) {
return new AttributeMapper<T>() {
@Override
public String name() {
return attributeName;
}
@Override
public T readAttribute(AttributedElement enclosing, ClassReader cf, int pos) {
return attributeFactory.apply(this, cf.readBytes(pos, cf.readInt(pos - 4)));
}
@Override
public void writeAttribute(BufWriter buf, T attr) {
buf.writeBytes(contentAccessor.apply(attr));
}
};
}
static AttributeMapper<? extends CustomAttribute> selfContainedCustomAttribute(String attributeName) {
class SelfContainedAttribute extends CustomAttribute<SelfContainedAttribute> {
final byte[] content;
public SelfContainedAttribute(AttributeMapper<SelfContainedAttribute> mapper, byte[] content) {
super(mapper);
this.content = content;
}
}
return selfContainedCustomAttribute(attributeName, SelfContainedAttribute::new, a -> a.content);
}
For category #2 there might be more options:
* A factory producing mapper which throws on write when CP is not shared
This is default behavior of unknown attributes, so there is no user action needed.
* Or a factory producing mapper simplifying CP entries clone and re-mapping on write when CP is not shared:
static <T extends CustomAttribute<T>>
AttributeMapper<T> singleConstantPoolEntryCustomAttribute(String attributeName,
BiFunction<AttributeMapper<T>, PoolEntry, T> attributeFactory,
Function<T, PoolEntry> entryAccessor) {
return new AttributeMapper<T>() {
@Override
public String name() {
return attributeName;
}
@Override
public T readAttribute(AttributedElement enclosing, ClassReader cf, int pos) {
return attributeFactory.apply(this, cf.readEntryOrNull(pos));
}
@Override
public void writeAttribute(BufWriter buf, T attr) {
buf.writeIndexOrZero(entryAccessor.apply(attr));
}
};
}
static AttributeMapper<? extends CustomAttribute> singleConstantPoolEntryCustomAttribute(String attributeName) {
class SingleEntryAttribute extends CustomAttribute<SingleEntryAttribute> {
final PoolEntry entry;
public SingleEntryAttribute(AttributeMapper<SingleEntryAttribute> mapper, PoolEntry entry) {
super(mapper);
this.entry = entry;
}
}
return singleConstantPoolEntryCustomAttribute(attributeName, SingleEntryAttribute::new, a -> a.entry);
}
-------------- next part --------------
An HTML attachment was scrubbed...
URL: <https://mail.openjdk.org/pipermail/classfile-api-dev/attachments/20230731/488654e7/attachment-0001.htm>
More information about the classfile-api-dev
mailing list