blob: 3855c3f9bf626cbf7c55b2bf8e9df427b4347660 [file] [log] [blame]
{%- from "constant_definition.tmpl" import constant_def %}
{%- from "enum_definition.tmpl" import enum_def %}
{%- macro equality(kind, v1, v2, ne=False) -%}
{%- if kind|is_reference_kind -%}
{%- if kind|is_array_kind -%}
{%- if kind.kind|is_reference_kind -%}
{%- if ne %}!{%- endif %}java.util.Arrays.deepEquals({{v1}}, {{v2}})
{%- else -%}
{%- if ne %}!{%- endif %}java.util.Arrays.equals({{v1}}, {{v2}})
{%- endif -%}
{%- else -%}
{%- if ne %}!{%- endif %}org.chromium.mojo.bindings.BindingsHelper.equals({{v1}}, {{v2}})
{%- endif -%}
{%- else -%}
{{v1}} {%- if ne %}!={%- else %}=={%- endif %} {{v2}}
{%- endif -%}
{%- endmacro -%}
{%- macro hash(kind, v) -%}
{%- if kind|is_array_kind -%}
{%- if kind.kind|is_reference_kind -%}
java.util.Arrays.deepHashCode({{v}})
{%- else -%}
java.util.Arrays.hashCode({{v}})
{%- endif -%}
{%- else -%}
org.chromium.mojo.bindings.BindingsHelper.hashCode({{v}})
{%- endif -%}
{%- endmacro -%}
{%- macro array_element_size(kind) -%}
{%- if kind|is_union_kind %}
org.chromium.mojo.bindings.BindingsHelper.UNION_SIZE
{%- else -%}
org.chromium.mojo.bindings.BindingsHelper.POINTER_SIZE
{%- endif -%}
{%- endmacro -%}
{%- macro encode(variable, kind, offset, bit, level=0, check_for_null=True) %}
{%- if kind|is_pointer_array_kind or kind|is_union_array_kind %}
{%- set sub_kind = kind.kind %}
{%- if check_for_null %}
if ({{variable}} == null) {
encoder{{level}}.encodeNullPointer({{offset}}, {{kind|is_nullable_kind|java_true_false}});
} else {
{%- else %}
{
{%- endif %}
{%- if kind|is_pointer_array_kind %}
{%- set encodePointer = 'encodePointerArray' %}
{%- else %}
{%- set encodePointer = 'encodeUnionArray' %}
{%- endif %}
org.chromium.mojo.bindings.Encoder encoder{{level + 1}} = encoder{{level}}.{{encodePointer}}({{variable}}.length, {{offset}}, {{kind|array_expected_length}});
for (int i{{level}} = 0; i{{level}} < {{variable}}.length; ++i{{level}}) {
{{encode(variable~'[i'~level~']', sub_kind, 'org.chromium.mojo.bindings.DataHeader.HEADER_SIZE + ' ~ array_element_size(sub_kind) ~ ' * i'~level, 0, level+1)|indent(8)}}
}
}
{%- elif kind|is_map_kind %}
if ({{variable}} == null) {
encoder{{level}}.encodeNullPointer({{offset}}, {{kind|is_nullable_kind|java_true_false}});
} else {
org.chromium.mojo.bindings.Encoder encoder{{level + 1}} = encoder{{level}}.encoderForMap({{offset}});
int size{{level}} = {{variable}}.size();
{{kind.key_kind|java_type}}[] keys{{level}} = {{kind.key_kind|array|new_array('size'~level)}};
{{kind.value_kind|java_type}}[] values{{level}} = {{kind.value_kind|array|new_array('size'~level)}};
int index{{level}} = 0;
for (java.util.Map.Entry<{{kind.key_kind|java_type(true)}}, {{kind.value_kind|java_type(true)}}> entry{{level}} : {{variable}}.entrySet()) {
keys{{level}}[index{{level}}] = entry{{level}}.getKey();
values{{level}}[index{{level}}] = entry{{level}}.getValue();
++index{{level}};
}
{{encode('keys'~level, kind.key_kind|array, 'org.chromium.mojo.bindings.DataHeader.HEADER_SIZE', 0, level+1, False)|indent(4)}}
{{encode('values'~level, kind.value_kind|array, 'org.chromium.mojo.bindings.DataHeader.HEADER_SIZE + org.chromium.mojo.bindings.BindingsHelper.POINTER_SIZE', 0, level+1, False)|indent(4)}}
}
{%- else %}
encoder{{level}}.{{kind|encode_method(variable, offset, bit)}};
{%- endif %}
{%- endmacro %}
{%- macro decode(variable, kind, offset, bit, level=0) %}
{%- if kind|is_struct_kind or
kind|is_pointer_array_kind or
kind|is_union_array_kind or
kind|is_map_kind %}
org.chromium.mojo.bindings.Decoder decoder{{level+1}} = decoder{{level}}.readPointer({{offset}}, {{kind|is_nullable_kind|java_true_false}});
{%- if kind|is_struct_kind %}
{{variable}} = {{kind|java_type}}.decode(decoder{{level+1}});
{%- else %}{# kind|is_pointer_array_kind or is_map_kind #}
{%- if kind|is_nullable_kind %}
if (decoder{{level+1}} == null) {
{{variable}} = null;
} else {
{%- else %}
{
{%- endif %}
{%- if kind|is_map_kind %}
decoder{{level+1}}.readDataHeaderForMap();
{{kind.key_kind|java_type}}[] keys{{level}};
{{kind.value_kind|java_type}}[] values{{level}};
{
{{decode('keys'~level, kind.key_kind|array, 'org.chromium.mojo.bindings.DataHeader.HEADER_SIZE', 0, level+1)|indent(8)}}
}
{
{{decode('values'~level, kind.value_kind|array('keys'~level~'.length'), 'org.chromium.mojo.bindings.DataHeader.HEADER_SIZE + org.chromium.mojo.bindings.BindingsHelper.POINTER_SIZE', 0, level+1)|indent(8)}}
}
{{variable}} = new java.util.HashMap<{{kind.key_kind|java_type(true)}}, {{kind.value_kind|java_type(true)}}>();
for (int index{{level}} = 0; index{{level}} < keys{{level}}.length; ++index{{level}}) {
{{variable}}.put(keys{{level}}[index{{level}}], values{{level}}[index{{level}}]);
}
{%- else %}
org.chromium.mojo.bindings.DataHeader si{{level+1}} = decoder{{level+1}}.readDataHeaderForPointerArray({{kind|array_expected_length}});
{{variable}} = {{kind|new_array('si'~(level+1)~'.elementsOrVersion')}};
for (int i{{level+1}} = 0; i{{level+1}} < si{{level+1}}.elementsOrVersion; ++i{{level+1}}) {
{{decode(variable~'[i'~(level+1)~']', kind.kind, 'org.chromium.mojo.bindings.DataHeader.HEADER_SIZE + ' ~ array_element_size(kind.kind) ~' * i'~(level+1), 0, level+1)|indent(8)}}
}
{%- endif %}
}
{%- endif %}
{%- elif kind|is_union_kind %}
{{variable}} = {{kind|java_type}}.decode(decoder{{level}}, {{offset}});
{%- else %}
{{variable}} = decoder{{level}}.{{kind|decode_method(offset, bit)}};
{%- if kind|is_array_kind and kind.kind|is_enum_kind %}
{%- if kind|is_nullable_kind %}
if ({{variable}} != null) {
{%- else %}
{
{%- endif %}
for (int i{{level}} = 0; i{{level}} < {{variable}}.length; ++i{{level}}) {
{{kind.kind|java_class_for_enum}}.validate({{variable}}[i{{level}}]);
}
}
{%- elif kind|is_enum_kind %}
{{kind|java_class_for_enum}}.validate({{variable}});
{%- endif %}
{%- endif %}
{%- endmacro %}
{%- macro struct_def(struct, inner_class=False) %}
{{'static' if inner_class else 'public'}} final class {{struct|name}} extends org.chromium.mojo.bindings.Struct {
private static final int STRUCT_SIZE = {{struct.versions[-1].num_bytes}};
private static final org.chromium.mojo.bindings.DataHeader[] VERSION_ARRAY = new org.chromium.mojo.bindings.DataHeader[] {
{%- for version in struct.versions -%}
new org.chromium.mojo.bindings.DataHeader({{version.num_bytes}}, {{version.version}}){%- if not loop.last %}, {%- endif -%}
{%- endfor -%}
};
private static final org.chromium.mojo.bindings.DataHeader DEFAULT_STRUCT_INFO = VERSION_ARRAY[{{struct.versions|length - 1}}];
{%- for constant in struct.constants %}
{{constant_def(constant)|indent(4)}}
{%- endfor %}
{%- for enum in struct.enums %}
{{enum_def(enum, false)|indent(4)}}
{%- endfor %}
{%- if struct.fields %}
{%- for field in struct.fields %}
public {{field.kind|java_type}} {{field|name}};
{%- endfor %}
{%- endif %}
private {{struct|name}}(int version) {
super(STRUCT_SIZE, version);
{%- for field in struct.fields %}
{%- if field.default %}
this.{{field|name}} = {{field|default_value}};
{%- elif field.kind|is_any_handle_kind %}
this.{{field|name}} = org.chromium.mojo.system.InvalidHandle.INSTANCE;
{%- endif %}
{%- endfor %}
}
public {{struct|name}}() {
this({{struct.versions[-1].version}});
}
public static {{struct|name}} deserialize(org.chromium.mojo.bindings.Message message) {
return decode(new org.chromium.mojo.bindings.Decoder(message));
}
/**
* Similar to the method above, but deserializes from a |ByteBuffer| instance.
*
* @throws org.chromium.mojo.bindings.DeserializationException on deserialization failure.
*/
public static {{struct|name}} deserialize(java.nio.ByteBuffer data) {
if (data == null)
return null;
return deserialize(new org.chromium.mojo.bindings.Message(
data, new java.util.ArrayList<org.chromium.mojo.system.Handle>()));
}
@SuppressWarnings("unchecked")
public static {{struct|name}} decode(org.chromium.mojo.bindings.Decoder decoder0) {
if (decoder0 == null) {
return null;
}
decoder0.increaseStackDepth();
{{struct|name}} result;
try {
org.chromium.mojo.bindings.DataHeader mainDataHeader = decoder0.readAndValidateDataHeader(VERSION_ARRAY);
result = new {{struct|name}}(mainDataHeader.elementsOrVersion);
{%- for byte in struct.bytes %}
{%- for packed_field in byte.packed_fields %}
if (mainDataHeader.elementsOrVersion >= {{packed_field.min_version}}) {
{{decode('result.' ~ packed_field.field|name, packed_field.field.kind, 8+packed_field.offset, packed_field.bit)|indent(16)}}
}
{%- endfor %}
{%- endfor %}
} finally {
decoder0.decreaseStackDepth();
}
return result;
}
@SuppressWarnings("unchecked")
@Override
protected final void encode(org.chromium.mojo.bindings.Encoder encoder) {
{%- if not struct.bytes %}
encoder.getEncoderAtDataOffset(DEFAULT_STRUCT_INFO);
{%- else %}
org.chromium.mojo.bindings.Encoder encoder0 = encoder.getEncoderAtDataOffset(DEFAULT_STRUCT_INFO);
{%- endif %}
{%- for byte in struct.bytes %}
{%- for packed_field in byte.packed_fields %}
{{encode('this.' ~ packed_field.field|name, packed_field.field.kind, 8+packed_field.offset, packed_field.bit)|indent(8)}}
{%- endfor %}
{%- endfor %}
}
/**
* @see Object#equals(Object)
*/
@Override
public boolean equals(Object object) {
if (object == this)
return true;
if (object == null)
return false;
if (getClass() != object.getClass())
return false;
{%- if struct.fields|length %}
{{struct|name}} other = ({{struct|name}}) object;
{%- for field in struct.fields %}
if ({{equality(field.kind, 'this.'~field|name, 'other.'~field|name, True)}})
return false;
{%- endfor %}
{%- endif %}
return true;
}
/**
* @see Object#hashCode()
*/
@Override
public int hashCode() {
final int prime = 31;
int result = prime + getClass().hashCode();
{%- for field in struct.fields %}
result = prime * result + {{hash(field.kind, 'this.'~field|name)}};
{%- endfor %}
return result;
}
}
{%- endmacro %}
{%- macro union_def(union) %}
public final class {{union|name}} extends org.chromium.mojo.bindings.Union {
public static final class Tag {
{%- for field in union.fields %}
public static final int {{field|ucc}} = {{loop.index0}};
{%- endfor %}
};
private int mTag_ = -1;
{%- for field in union.fields %}
private {{field.kind|java_type}} m{{field|ucc}};
{%- endfor %}
public int which() {
return mTag_;
}
public boolean isUnknown() {
return mTag_ == -1;
}
{%- for field in union.fields %}
public void set{{field|ucc}}({{field.kind|java_type}} {{field|name}}) {
this.mTag_ = Tag.{{field|ucc}};
this.m{{field|ucc}} = {{field|name}};
}
public {{field.kind|java_type}} get{{field|ucc}}() {
assert this.mTag_ == Tag.{{field|ucc}};
return this.m{{field|ucc}};
}
{%- endfor %}
@Override
protected final void encode(org.chromium.mojo.bindings.Encoder encoder0, int offset) {
encoder0.encode(org.chromium.mojo.bindings.BindingsHelper.UNION_SIZE, offset);
encoder0.encode(this.mTag_, offset + 4);
switch (mTag_) {
{%- for field in union.fields %}
case Tag.{{field|ucc}}: {
{%- if field.kind|is_union_kind %}
if (this.m{{field|ucc}} == null) {
encoder0.encodeNullPointer(offset + 8, {{field.kind|is_nullable_kind|java_true_false}});
} else {
encoder0.encoderForUnionPointer(offset + 8).encode(
this.m{{field|ucc}}, 0, {{field.kind|is_nullable_kind|java_true_false}});
}
{%- else %}
{{encode('this.m' ~ field|ucc, field.kind, 'offset + 8', 0)|indent(16)}}
{%- endif %}
break;
}
{%- endfor %}
default: {
break;
}
}
}
public static {{union|name}} deserialize(org.chromium.mojo.bindings.Message message) {
return decode(new org.chromium.mojo.bindings.Decoder(message).decoderForSerializedUnion(), 0);
}
public static final {{union|name}} decode(org.chromium.mojo.bindings.Decoder decoder0, int offset) {
org.chromium.mojo.bindings.DataHeader dataHeader = decoder0.readDataHeaderForUnion(offset);
if (dataHeader.size == 0) {
return null;
}
{{union|name}} result = new {{union|name}}();
switch (dataHeader.elementsOrVersion) {
{%- for field in union.fields %}
case Tag.{{field|ucc}}: {
{%- if field.kind|is_union_kind %}
org.chromium.mojo.bindings.Decoder decoder1 = decoder0.readPointer(offset + org.chromium.mojo.bindings.DataHeader.HEADER_SIZE, {{field.kind|is_nullable_kind|java_true_false}});
if (decoder1 != null) {
result.m{{field|ucc}} = {{field.kind|java_type}}.decode(decoder1, 0);
}
{%- else %}
{{decode('result.m'~field|ucc, field.kind, 'offset + org.chromium.mojo.bindings.DataHeader.HEADER_SIZE', 0)|indent(16)}}
{%- endif %}
result.mTag_ = Tag.{{field|ucc}};
break;
}
{%- endfor %}
default: {
break;
}
}
return result;
}
/**
* @see Object#equals(Object)
*/
@Override
public boolean equals(Object object) {
if (object == this)
return true;
if (object == null)
return false;
if (getClass() != object.getClass())
return false;
{{union|name}} other = ({{union|name}}) object;
if (this.mTag_ != other.mTag_)
return false;
switch (this.mTag_) {
{%- for field in union.fields %}
case Tag.{{field|ucc}}:
return {{equality(field.kind, 'this.m'~field|ucc, 'other.m'~field|ucc)}};
{%- endfor %}
default:
break;
}
return false;
}
/**
* @see Object#hashCode()
*/
@Override
public int hashCode() {
final int prime = 31;
int result = prime + getClass().hashCode();
result = prime * result + org.chromium.mojo.bindings.BindingsHelper.hashCode(mTag_);
switch (this.mTag_) {
{%- for field in union.fields %}
case Tag.{{field|ucc}}: {
result = prime * result + {{hash(field.kind, 'this.m'~field|ucc)}};
break;
}
{%- endfor %}
default: {
break;
}
}
return result;
}
}
{%- endmacro %}