1 /*******************************************************************************
2 * Copyright (c) 2011, 2013 Ericsson, École Polytechnique de Montréal
4 * All rights reserved. This program and the accompanying materials are made
5 * available under the terms of the Eclipse Public License v1.0 which
6 * accompanies this distribution, and is available at
7 * http://www.eclipse.org/legal/epl-v10.html
10 * Matthew Khouzam - Initial API and implementation
11 * Alexandre Montplaisir - Initial API and implementation, extend TmfEventField
12 * Bernd Hufmann - Add Enum field handling
13 * Geneviève Bastien - Add Struct and Variant field handling
14 * Jean-Christian Kouame - Correct handling of unsigned integer fields
15 *******************************************************************************/
17 package org
.eclipse
.linuxtools
.tmf
.core
.ctfadaptor
;
19 import java
.util
.ArrayList
;
20 import java
.util
.Arrays
;
21 import java
.util
.List
;
22 import java
.util
.Map
.Entry
;
24 import org
.eclipse
.linuxtools
.ctf
.core
.event
.types
.ArrayDeclaration
;
25 import org
.eclipse
.linuxtools
.ctf
.core
.event
.types
.ArrayDefinition
;
26 import org
.eclipse
.linuxtools
.ctf
.core
.event
.types
.Definition
;
27 import org
.eclipse
.linuxtools
.ctf
.core
.event
.types
.EnumDefinition
;
28 import org
.eclipse
.linuxtools
.ctf
.core
.event
.types
.FloatDefinition
;
29 import org
.eclipse
.linuxtools
.ctf
.core
.event
.types
.IntegerDeclaration
;
30 import org
.eclipse
.linuxtools
.ctf
.core
.event
.types
.IntegerDefinition
;
31 import org
.eclipse
.linuxtools
.ctf
.core
.event
.types
.SequenceDeclaration
;
32 import org
.eclipse
.linuxtools
.ctf
.core
.event
.types
.SequenceDefinition
;
33 import org
.eclipse
.linuxtools
.ctf
.core
.event
.types
.StringDefinition
;
34 import org
.eclipse
.linuxtools
.ctf
.core
.event
.types
.StructDefinition
;
35 import org
.eclipse
.linuxtools
.ctf
.core
.event
.types
.VariantDefinition
;
36 import org
.eclipse
.linuxtools
.internal
.tmf
.core
.Messages
;
37 import org
.eclipse
.linuxtools
.tmf
.core
.event
.ITmfEventField
;
38 import org
.eclipse
.linuxtools
.tmf
.core
.event
.TmfEventField
;
41 * The CTF implementation of the TMF event field model
44 * @author Matthew Khouzam
45 * @author Alexandre Montplaisir
47 public abstract class CtfTmfEventField
extends TmfEventField
{
49 // ------------------------------------------------------------------------
51 // ------------------------------------------------------------------------
54 * Standard constructor. Only to be used internally, call parseField() to
55 * generate a new field object.
58 * The name of this field
60 * The value of this field. Its type should match the field type.
62 * The children fields. Useful for composite fields
65 protected CtfTmfEventField(String name
, Object value
, ITmfEventField
[] fields
) {
66 super(/* Strip the underscore from the field name if there is one */
67 name
.startsWith("_") ? name
.substring(1) : name
, //$NON-NLS-1$
72 // ------------------------------------------------------------------------
74 // ------------------------------------------------------------------------
77 * Factory method to instantiate CtfTmfEventField objects.
80 * The CTF Definition of this event field
82 * String The name to assign to this field
83 * @return The resulting CtfTmfEventField object
85 public static CtfTmfEventField
parseField(Definition fieldDef
,
87 CtfTmfEventField field
= null;
89 /* Determine the Definition type */
90 if (fieldDef
instanceof IntegerDefinition
) {
91 IntegerDefinition intDef
= (IntegerDefinition
) fieldDef
;
92 int base
= intDef
.getDeclaration().getBase();
93 field
= new CTFIntegerField(fieldName
, intDef
.getValue(), base
, intDef
.getDeclaration().isSigned());
95 } else if (fieldDef
instanceof EnumDefinition
) {
96 EnumDefinition enumDef
= (EnumDefinition
) fieldDef
;
97 field
= new CTFEnumField(fieldName
, new CtfEnumPair(enumDef
.getValue(), enumDef
.getIntegerValue()));
99 } else if (fieldDef
instanceof StringDefinition
) {
100 field
= new CTFStringField(fieldName
, ((StringDefinition
) fieldDef
).getValue());
102 } else if (fieldDef
instanceof FloatDefinition
) {
103 FloatDefinition floatDef
= (FloatDefinition
) fieldDef
;
104 field
= new CTFFloatField(fieldName
, floatDef
.getValue());
106 } else if (fieldDef
instanceof ArrayDefinition
) {
107 ArrayDefinition arrayDef
= (ArrayDefinition
) fieldDef
;
108 ArrayDeclaration arrayDecl
= arrayDef
.getDeclaration();
110 if (arrayDef
.isString()) {
111 /* This is an array of UTF-8 bytes, a.k.a. a String! */
112 field
= new CTFStringField(fieldName
, fieldDef
.toString());
114 } else if (arrayDecl
.getElementType() instanceof IntegerDeclaration
) {
115 /* This is a an array of CTF Integers */
116 List
<Long
> values
= new ArrayList
<Long
>(arrayDecl
.getLength());
117 for (int i
= 0; i
< arrayDecl
.getLength(); i
++) {
118 values
.add(((IntegerDefinition
) arrayDef
.getElem(i
)).getValue());
120 long[] valuesArray
= convertListToArray(values
);
121 field
= new CTFIntegerArrayField(fieldName
, valuesArray
,
122 ((IntegerDeclaration
) arrayDecl
.getElementType()).getBase(),
123 ((IntegerDeclaration
) arrayDecl
.getElementType()).isSigned());
125 /* Add other types of arrays here */
127 } else if (fieldDef
instanceof SequenceDefinition
) {
128 SequenceDefinition seqDef
= (SequenceDefinition
) fieldDef
;
129 SequenceDeclaration seqDecl
= seqDef
.getDeclaration();
131 if (seqDef
.getLength() == 0) {
132 /* Some sequences have length = 0. Simply use an empty string */
133 field
= new CTFStringField(fieldName
, ""); //$NON-NLS-1$
134 } else if (seqDef
.isString()) {
135 /* Interpret this sequence as a String */
136 field
= new CTFStringField(fieldName
, seqDef
.toString());
137 } else if (seqDecl
.getElementType() instanceof IntegerDeclaration
) {
138 /* Sequence of integers => CTFIntegerArrayField */
139 List
<Long
> values
= new ArrayList
<Long
>(seqDef
.getLength());
140 for (int i
= 0; i
< seqDef
.getLength(); i
++) {
141 values
.add(((IntegerDefinition
) seqDef
.getElem(i
)).getValue());
143 long[] valuesArray
= convertListToArray(values
);
144 field
= new CTFIntegerArrayField(fieldName
, valuesArray
,
145 ((IntegerDeclaration
) seqDecl
.getElementType()).getBase(),
146 ((IntegerDeclaration
) seqDecl
.getElementType()).isSigned());
148 /* Add other Sequence types here */
150 } else if (fieldDef
instanceof StructDefinition
) {
151 StructDefinition strDef
= (StructDefinition
) fieldDef
;
153 String curFieldName
= null;
154 Definition curFieldDef
;
155 CtfTmfEventField curField
;
156 List
<ITmfEventField
> list
= new ArrayList
<ITmfEventField
>();
157 /* Recursively parse the fields */
158 for (Entry
<String
, Definition
> entry
: strDef
.getDefinitions().entrySet()) {
159 curFieldName
= entry
.getKey();
160 curFieldDef
= entry
.getValue();
161 curField
= CtfTmfEventField
.parseField(curFieldDef
, curFieldName
);
164 field
= new CTFStructField(fieldName
, list
.toArray(new CtfTmfEventField
[list
.size()]));
166 } else if (fieldDef
instanceof VariantDefinition
) {
167 VariantDefinition varDef
= (VariantDefinition
) fieldDef
;
169 String curFieldName
= varDef
.getCurrentFieldName();
170 Definition curFieldDef
= varDef
.getDefinitions().get(curFieldName
);
171 if (curFieldDef
!= null) {
172 CtfTmfEventField subField
= CtfTmfEventField
.parseField(curFieldDef
, curFieldName
);
173 field
= new CTFVariantField(fieldName
, subField
);
175 /* A safe-guard, but curFieldDef should never be null */
176 field
= new CTFStringField(curFieldName
, ""); //$NON-NLS-1$
181 * Safe-guard, to avoid null exceptions later, field is expected not
184 field
= new CTFStringField(fieldName
, Messages
.TmfEventField_UnsupportedType
+ fieldDef
.getClass().toString());
190 public String
toString() {
191 return getName() + '=' + getFormattedValue();
195 * We cannot use List.toArray(T[]) for primitives types, so do it manually.
197 private static long[] convertListToArray(List
<Long
> list
) {
198 long[] array
= new long[list
.size()];
199 for (int i
= 0; i
< list
.size(); i
++) {
200 array
[i
] = list
.get(i
);
208 * The CTF field implementation for integer fields.
212 final class CTFIntegerField
extends CtfTmfEventField
{
214 private final int base
;
215 private final boolean signed
;
218 * A CTF "IntegerDefinition" can be an integer of any byte size, so in the
219 * Java parser this is interpreted as a long.
222 * The name of this field
224 * The integer value of this field
226 * Is the value signed or not
228 CTFIntegerField(String name
, long longValue
, int base
, boolean signed
) {
229 super(name
, longValue
, null);
230 this.signed
= signed
;
235 public Long
getValue() {
236 return (Long
) super.getValue();
240 public String
getFormattedValue() {
241 return IntegerDefinition
.formatNumber(getValue(), base
, signed
);
247 * The CTF field implementation for string fields
251 final class CTFStringField
extends CtfTmfEventField
{
254 * Constructor for CTFStringField.
257 * The string value of this field
259 * The name of this field
261 CTFStringField(String name
, String strValue
) {
262 super(name
, strValue
, null);
266 public String
getValue() {
267 return (String
) super.getValue();
272 * CTF field implementation for arrays of integers.
276 final class CTFIntegerArrayField
extends CtfTmfEventField
{
278 private final int base
;
279 private final boolean signed
;
280 private String formattedValue
= null;
283 * Constructor for CTFIntegerArrayField.
286 * The name of this field
288 * The array of integers (as longs) that compose this field's
291 * Are the values in the array signed or not
293 CTFIntegerArrayField(String name
, long[] longValues
, int base
, boolean signed
) {
294 super(name
, longValues
, null);
296 this.signed
= signed
;
300 public long[] getValue() {
301 return (long[]) super.getValue();
305 public String
getFormattedValue() {
306 if (formattedValue
== null) {
307 List
<String
> strings
= new ArrayList
<String
>();
308 for (long value
: getValue()) {
309 strings
.add(IntegerDefinition
.formatNumber(value
, base
, signed
));
311 formattedValue
= strings
.toString();
313 return formattedValue
;
319 * CTF field implementation for floats.
323 final class CTFFloatField
extends CtfTmfEventField
{
326 * Constructor for CTFFloatField.
329 * The float value (actually a double) of this field
331 * The name of this field
333 protected CTFFloatField(String name
, double value
) {
334 super(name
, value
, null);
338 public Double
getValue() {
339 return (Double
) super.getValue();
344 * The CTF field implementation for Enum fields
346 * @author Bernd Hufmann
348 final class CTFEnumField
extends CtfTmfEventField
{
351 * Constructor for CTFEnumField.
354 * The Enum value consisting of a pair of Enum value name and its
357 * The name of this field
359 CTFEnumField(String name
, CtfEnumPair enumValue
) {
360 super(name
, new CtfEnumPair(enumValue
.getFirst(),
361 enumValue
.getSecond().longValue()), null);
365 public CtfEnumPair
getValue() {
366 return (CtfEnumPair
) super.getValue();
371 * The CTF field implementation for struct fields with sub-fields
375 final class CTFStructField
extends CtfTmfEventField
{
378 * Constructor for CTFStructField.
381 * The children of this field
383 * The name of this field
385 CTFStructField(String name
, CtfTmfEventField
[] fields
) {
386 super(name
, fields
, fields
);
390 public CtfTmfEventField
[] getValue() {
391 return (CtfTmfEventField
[]) super.getValue();
395 public String
getFormattedValue() {
396 return Arrays
.toString(getValue());
402 * The CTF field implementation for variant fields its child
406 final class CTFVariantField
extends CtfTmfEventField
{
409 * Constructor for CTFVariantField.
412 * The field selected for this variant
414 * The name of this field
416 CTFVariantField(String name
, CtfTmfEventField field
) {
417 super(name
, field
, new CtfTmfEventField
[] { field
});
421 public CtfTmfEventField
getValue() {
422 return (CtfTmfEventField
) super.getValue();
427 /* Implement other possible fields types here... */