@############################################### @# @# PX4 ROS compatible message source code @# generation for C++ @# @# EmPy template for generating .h files @# Based on the original template for ROS @# @############################################### @# Start of Template @# @# Context: @# - file_name_in (String) Source file @# - spec (msggen.MsgSpec) Parsed specification of the .msg file @# - md5sum (String) MD5Sum of the .msg specification @# - search_path (dict) search paths for genmsg @# - topics (List of String) multi-topic names @############################################### /**************************************************************************** * * Copyright (C) 2013-2016 PX4 Development Team. All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in * the documentation and/or other materials provided with the * distribution. * 3. Neither the name PX4 nor the names of its contributors may be * used to endorse or promote products derived from this software * without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN * ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE * POSSIBILITY OF SUCH DAMAGE. * ****************************************************************************/ /* Auto-generated by genmsg_cpp from file @file_name_in */ @{ import genmsg.msgs import gencpp uorb_struct = '%s_s'%spec.short_name topic_name = spec.short_name type_map = { 'int8': 'int8_t', 'int16': 'int16_t', 'int32': 'int32_t', 'int64': 'int64_t', 'uint8': 'uint8_t', 'uint16': 'uint16_t', 'uint32': 'uint32_t', 'uint64': 'uint64_t', 'float32': 'float', 'float64': 'double', 'bool': 'bool', 'char': 'char', } msgtype_size_map = { 'int8': 1, 'int16': 2, 'int32': 4, 'int64': 8, 'uint8': 1, 'uint16': 2, 'uint32': 4, 'uint64': 8, 'float32': 4, 'float64': 8, 'bool': 1, 'char': 1, } def convert_type(spec_type): bare_type = spec_type if '/' in spec_type: # removing prefix bare_type = (spec_type.split('/'))[1] msg_type, is_array, array_length = genmsg.msgs.parse_type(bare_type) c_type = msg_type if msg_type in type_map: c_type = type_map[msg_type] if is_array: return c_type + "[" + str(array_length) + "]" return c_type def bare_name(msg_type): bare = msg_type if '/' in msg_type: # removing prefix bare = (msg_type.split('/'))[1] # removing suffix return bare.split('[')[0] def sizeof_field_type(field): bare_name_str = bare_name(field.type) if bare_name_str in msgtype_size_map: return msgtype_size_map[bare_name_str] return 0 # this is for non-builtin types: sort them at the end def get_children_fields(base_type): (package, name) = genmsg.names.package_resource_name(base_type) tmp_msg_context = genmsg.msg_loader.MsgContext.create_default() spec_temp = genmsg.msg_loader.load_msg_by_type(tmp_msg_context, '%s/%s' %(package, name), search_path) sorted_fields = sorted(spec_temp.parsed_fields(), key=sizeof_field_type, reverse=True) return sorted_fields # Add padding fields before the embedded types, at the end and calculate the # struct size def add_padding_bytes(fields): struct_size = 8 # account for the timestamp align_to = 8 # this is always 8, because of the 64bit timestamp i = 0 padding_idx = 0 while i < len(fields): field = fields[i] if not field.is_header: a_pos = field.type.find('[') array_size = 1 if field.is_array: array_size = field.array_len if field.is_builtin: field.sizeof_field_type = sizeof_field_type(field) else: # embedded type: may need to add padding num_padding_bytes = align_to - (struct_size % align_to) if num_padding_bytes != align_to: padding_field = genmsg.Field('_padding'+str(padding_idx), 'uint8['+str(num_padding_bytes)+']') padding_idx += 1 padding_field.sizeof_field_type = 1 struct_size += num_padding_bytes fields.insert(i, padding_field) i += 1 children_fields = get_children_fields(field.base_type) field.sizeof_field_type, unused = add_padding_bytes(children_fields) struct_size += field.sizeof_field_type * array_size i += 1 # add padding at the end (necessary for embedded types) num_padding_bytes = align_to - (struct_size % align_to) if num_padding_bytes == align_to: num_padding_bytes = 0 else: padding_field = genmsg.Field('_padding'+str(padding_idx), 'uint8['+str(num_padding_bytes)+']') padding_idx += 1 padding_field.sizeof_field_type = 1 struct_size += num_padding_bytes fields.append(padding_field) return (struct_size, num_padding_bytes) sorted_fields = sorted(spec.parsed_fields(), key=sizeof_field_type, reverse=True) struct_size, padding_end_size = add_padding_bytes(sorted_fields) topic_fields = ["%s %s" % (convert_type(field.type), field.name) for field in sorted_fields] }@ #include #include #include @# join all msg files in one line e.g: "float[3] position;float[3] velocity;bool armed" const char *__orb_@(topic_name)_fields = "@( topic_name.upper() ):@( ";".join(topic_fields) );"; @[for multi_topic in topics]@ ORB_DEFINE(@multi_topic, struct @uorb_struct, @(struct_size-padding_end_size), __orb_@(topic_name)_fields); @[end for]