/* * Copyright (c) 2016, 2019, 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. Oracle designates this * particular file as subject to the "Classpath" exception as provided * by Oracle in the LICENSE file that accompanied this code. * * 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 jdk.jfr; import java.lang.annotation.Annotation; import java.util.ArrayList; import java.util.Collections; import java.util.List; import java.util.Objects; import jdk.jfr.internal.AnnotationConstruct; import jdk.jfr.internal.Type; import jdk.jfr.internal.Utils; /** * Describes the event fields and annotation elements. * * @since 8 */ public final class ValueDescriptor { private final AnnotationConstruct annotationConstruct; private final Type type; private final String name; private final boolean isArray; private final boolean constantPool; private final String javaFieldName; // package private, invoked by jdk.internal. ValueDescriptor(Type type, String name, List annotations, int dimension, boolean constantPool, String fieldName) { Objects.requireNonNull(annotations); if (dimension < 0) { throw new IllegalArgumentException("Dimension must be positive"); } this.name = Objects.requireNonNull(name, "Name of value descriptor can't be null"); this.type = Objects.requireNonNull(type); this.isArray = dimension > 0; this.constantPool = constantPool; this.annotationConstruct = new AnnotationConstruct(annotations); this.javaFieldName = fieldName; } /** *

* Constructs a value descriptor, useful for dynamically creating event types and * annotations. *

* The following types are supported: *

* *

* The name must be a valid Java identifier (for example, {@code "maxThroughput"}). See 3.8 * Java Language Specification for more information. * * @param type the type, not {@code null} * @param name the name, not {@code null} * * @throws SecurityException if a security manager is present and the caller * doesn't have {@code FlightRecorderPermission("registerEvent")} * */ public ValueDescriptor(Class type, String name) { this(type, name, Collections. emptyList()); } /** *

* Constructs a value descriptor, useful for dynamically creating event types and * annotations. *

* The following types are supported: *

* *

* The name must be a valid Java identifier (for example, {@code "maxThroughput"}). See 3.8 * Java Language Specification for more information. * * @param type the type, not {@code null} * @param name the name, not {@code null} * @param annotations the annotations on the value descriptors, not * {@code null} * * @throws SecurityException if a security manager is present and the caller * doesn't have {@code FlightRecorderPermission("registerEvent")} */ public ValueDescriptor(Class type, String name, List annotations) { this(type, name, new ArrayList<>(annotations), false); } ValueDescriptor(Class type, String name, List annotations, boolean allowArray) { Objects.requireNonNull(annotations); Utils.checkRegisterPermission(); if (!allowArray) { if (type.isArray()) { throw new IllegalArgumentException("Array types are not allowed"); } } this.name = Objects.requireNonNull(name, "Name of value descriptor can't be null"); this.type = Objects.requireNonNull(Utils.getValidType(Objects.requireNonNull(type), Objects.requireNonNull(name))); this.annotationConstruct = new AnnotationConstruct(annotations); this.javaFieldName = name; // Needed for dynamic events this.isArray = type.isArray(); // Assume we always want to store String and Thread in constant pool this.constantPool = type == Class.class || type == Thread.class; } /** * Returns a human-readable name that describes the value (for example, * {@code "Maximum Throughput"}). * * @return a human-readable name, or {@code null} if doesn't exist */ public String getLabel() { return annotationConstruct.getLabel(); } /** * Returns the name of the value (for example, {@code "maxThroughput"}). * * @return the name, not {@code null} */ public String getName() { return name; } /** * Returns a sentence describing the value (for example, {@code "Maximum * throughput in the transaction system. Value is reset after each new * batch."}). * * @return the description, or {@code null} if doesn't exist */ public String getDescription() { return annotationConstruct.getDescription(); } /** * Returns a textual identifier that specifies how a value represented by * this {@link ValueDescriptor} is interpreted or formatted. *

* For example, if the value descriptor's type is {@code float} and the * event value is {@code 0.5f}, a content type of * {@code "jdk.jfr.Percentage"} hints to a client that the value is a * percentage and that it should be rendered as {@code "50%"}. *

* The JDK provides the following predefined content types: *

*

* User-defined content types can be created by using the {@link ContentType} class. * * @return the content type, or {@code null} if doesn't exist * * @see ContentType */ public String getContentType() { for (AnnotationElement anno : getAnnotationElements()) { for (AnnotationElement meta : anno.getAnnotationElements()) { if (meta.getTypeName().equals(ContentType.class.getName())) { return anno.getTypeName(); } } } return null; } /** * Returns the fully qualified class name of the type that is associated with * this value descriptor. * * @return the type name, not {@code null} * * @see ValueDescriptor#getTypeId() */ public String getTypeName() { if (type.isSimpleType()) { return type.getFields().get(0).getTypeName(); } return type.getName(); } /** * Returns a unique ID for the type in the Java virtual Machine (JVM). * * The ID might not be the same between JVM instances. * * @return the type ID, not negative */ public long getTypeId() { return type.getId(); } /** * Returns if this value descriptor is an array type. * * @return {@code true} if it is an array type, {@code false} otherwise */ public boolean isArray() { return isArray; } /** * Returns the first annotation for the specified type if an annotation * element with the same name is directly present for this value descriptor, * {@code null} otherwise. * * @param the type of the annotation to query for and return if present * @param annotationType the Class object that corresponds to the annotation * type, not {@code null} * @return this element's annotation for the specified annotation type if * directly present, else {@code null} */ public A getAnnotation(Class annotationType) { Objects.requireNonNull(annotationType); return annotationConstruct.getAnnotation(annotationType); } /** * Returns an immutable list of annotation elements for this value * descriptor. * * @return a list of annotations, not {@code null} */ public List getAnnotationElements() { return annotationConstruct.getUnmodifiableAnnotationElements(); } /** * Returns an immutable list of value descriptors if the type is complex, * else an empty list. * * @return a list of value descriptors, not {@code null} */ public List getFields() { if (type.isSimpleType()) { return Collections.emptyList(); } return type.getFields(); } // package private Type getType() { return type; } // package private void setAnnotations(List anno) { annotationConstruct.setAnnotationElements(anno); } // package private boolean isConstantPool() { return constantPool; } // package private String getJavaFieldName() { return javaFieldName; } // package private boolean isUnsigned() { return annotationConstruct.hasUnsigned(); } }