/* Simulator connector for Airsim: https://github.com/Microsoft/AirSim */ #pragma once #include #include "SIM_Aircraft.h" namespace SITL { /* Airsim Simulator */ class AirSim : public Aircraft { public: AirSim(const char *home_str, const char *frame_str); /* update model by one time step */ void update(const struct sitl_input &input) override; /* static object creator */ static Aircraft *create(const char *home_str, const char *frame_str) { return new AirSim(home_str, frame_str); } /* Create and set in/out socket for Airsim simulator */ void set_interface_ports(const char* address, const int port_in, const int port_out) override; private: /* rotor control packet sent by Ardupilot */ static const int kArduCopterRotorControlCount = 11; struct servo_packet { uint16_t pwm[kArduCopterRotorControlCount]; }; // default connection_info_.ip_address const char *airsim_ip = "127.0.0.1"; // connection_info_.ip_port uint16_t airsim_sensor_port = 9003; // connection_info_.sitl_ip_port uint16_t airsim_control_port = 9002; SocketAPM sock; double average_frame_time; uint64_t frame_counter; uint64_t last_frame_count; void send_servos(const struct sitl_input &input); void recv_fdm(); void report_FPS(void); bool parse_sensors(const char *json); // buffer for parsing pose data in JSON format uint8_t sensor_buffer[65000]; uint32_t sensor_buffer_len; enum data_type { DATA_UINT64, DATA_FLOAT, DATA_DOUBLE, DATA_VECTOR3F, DATA_VECTOR3F_ARRAY, DATA_FLOAT_ARRAY, }; struct { uint64_t timestamp; struct { Vector3f angular_velocity; Vector3f linear_acceleration; } imu; struct { float lat, lon, alt; } gps; struct { float roll, pitch, yaw; } pose; struct { Vector3f world_linear_velocity; } velocity; struct { struct vector3f_array points; } lidar; struct { struct float_array rc_channels; } rc; } state, last_state; // table to aid parsing of JSON sensor data struct keytable { const char *section; const char *key; void *ptr; enum data_type type; } keytable[12] = { { "", "timestamp", &state.timestamp, DATA_UINT64 }, { "imu", "angular_velocity", &state.imu.angular_velocity, DATA_VECTOR3F }, { "imu", "linear_acceleration", &state.imu.linear_acceleration, DATA_VECTOR3F }, { "gps", "lat", &state.gps.lat, DATA_FLOAT }, { "gps", "lon", &state.gps.lon, DATA_FLOAT }, { "gps", "alt", &state.gps.alt, DATA_FLOAT }, { "pose", "roll", &state.pose.roll, DATA_FLOAT }, { "pose", "pitch", &state.pose.pitch, DATA_FLOAT }, { "pose", "yaw", &state.pose.yaw, DATA_FLOAT }, { "velocity", "world_linear_velocity", &state.velocity.world_linear_velocity, DATA_VECTOR3F }, { "lidar", "point_cloud", &state.lidar.points, DATA_VECTOR3F_ARRAY }, { "rc", "channels", &state.rc.rc_channels, DATA_FLOAT_ARRAY }, }; }; }