Lap counter peripheral added
[apps/native/gear-racing-car.git] / src / app.c
1 /*
2  * Copyright (c) 2017 Samsung Electronics Co., Ltd.
3  *
4  * Contact: Jeonghoon Park <jh1979.park@samsung.com>
5  *
6  * Licensed under the Flora License, Version 1.1 (the License);
7  * you may not use this file except in compliance with the License.
8  * You may obtain a copy of the License at
9  *
10  * http://floralicense.org/license/
11  *
12  * Unless required by applicable law or agreed to in writing, software
13  * distributed under the License is distributed on an AS IS BASIS,
14  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
15  * See the License for the specific language governing permissions and
16  * limitations under the License.
17  */
18
19 #include <stdio.h>
20 #include <stdlib.h>
21 #include <unistd.h>
22 #include <math.h>
23 #include <glib.h>
24 #include <service_app.h>
25 #include "log.h"
26 #include "resource.h"
27 #include "net-util.h"
28 #include "config.h"
29 #include "cloud/cloud_communication.h"
30 #include "messages/message_manager.h"
31 #include "controller_connection_manager.h"
32 #include "lap_counter/lap_counter.h"
33 #include "command.h"
34
35 #define ENABLE_MOTOR 1
36 #define STERING_SERVO_CENTER 340
37 #define STERING_SERVO_RANGE 125
38
39 #define MAX_UDP_INPUT 10000
40
41 #define CONFIG_GRP_CAR "Car"
42 #define CONFIG_KEY_ID "Id"
43 #define CONFIG_KEY_NAME "Name"
44 #define CLOUD_REQUESTS_FREQUENCY 15
45 #define AZIMUTH_SERVO_PIN 15
46 #define ELEVATION_SERVO_PIN 14
47
48 #define ELEVATION_MIN 200
49 #define ELEVATION_MAX 400
50 #define AZIMUTH_MIN 200
51 #define AZIMUTH_MAX 700
52
53 enum {
54         DIR_STATE_S,
55         DIR_STATE_F,
56         DIR_STATE_B,
57 };
58
59 typedef struct app_data_s {
60         unsigned int f_value;
61         unsigned int r_value;
62         unsigned int dir_state;
63         const char *user_name;
64         guint idle_h;
65 } app_data;
66
67 static void _initialize_components(app_data *ad);
68 static void _initialize_config();
69
70 static void service_app_lang_changed(app_event_info_h event_info, void *user_data)
71 {
72         return;
73 }
74
75 static void service_app_region_changed(app_event_info_h event_info, void *user_data)
76 {
77         return;
78 }
79
80 static void service_app_low_battery(app_event_info_h event_info, void *user_data)
81 {
82         _E("low battery! exit app");
83         service_app_exit();
84
85         return;
86 }
87
88 static void service_app_low_memory(app_event_info_h event_info, void *user_data)
89 {
90         return;
91 }
92
93 static inline double __map_round(double val)
94 {
95         return floor(val + 0.5);
96 }
97
98 static int __map_range_val(int v_min, int v_max, int d_min, int d_max, int val)
99 {
100         int rval = 0;
101         double slope = 0;
102         slope = 1.0 * (d_max - d_min) / (v_max - v_min);
103
104         rval = d_min + __map_round(slope * (val - v_min));
105
106         return rval;
107 }
108
109 static inline int ___map_speed_val(int speed)
110 {
111         return __map_range_val(-MAX_UDP_INPUT, MAX_UDP_INPUT, -4095, 4095, speed);
112 }
113
114 static inline int ___map_servo_val(int servo)
115 {
116         return __map_range_val(-MAX_UDP_INPUT, MAX_UDP_INPUT, STERING_SERVO_CENTER - STERING_SERVO_RANGE, STERING_SERVO_CENTER + STERING_SERVO_RANGE, servo);
117 }
118
119 static int __driving_motors(int servo, int speed)
120 {
121         int val_speed;
122         int val_servo;
123
124         val_servo = ___map_servo_val(servo);
125         val_speed = ___map_speed_val(speed);
126
127         _D("control motor - servo[%4d : %4d], speed[%4d : %4d]",
128                 servo, val_servo, speed, val_speed);
129 #if ENABLE_MOTOR
130         resource_set_servo_motor_value(0, val_servo);
131         resource_set_motor_driver_L298N_speed(MOTOR_ID_1, val_speed);
132         resource_set_motor_driver_L298N_speed(MOTOR_ID_2, val_speed);
133 #endif
134
135         return 0;
136 }
137
138 static void __camera(int azimuth, int elevation)
139 {
140         int val_azimuth = __map_range_val(-MAX_UDP_INPUT, MAX_UDP_INPUT, AZIMUTH_MIN, AZIMUTH_MAX, azimuth);
141         int val_elevation = __map_range_val(0, MAX_UDP_INPUT, ELEVATION_MIN, ELEVATION_MAX, elevation); // No need to look upside down
142
143         _D("camera - azimuth[%4d : %4d], elevation[%4d : %4d]", azimuth, val_azimuth, elevation, val_elevation);
144 #if ENABLE_MOTOR
145         resource_set_servo_motor_value(ELEVATION_SERVO_PIN, val_elevation);
146         resource_set_servo_motor_value(AZIMUTH_SERVO_PIN, val_azimuth);
147 #endif
148 }
149
150 static void __command_received_cb(command_s command) {
151         switch(command.type) {
152         case COMMAND_TYPE_DRIVE:
153                 __driving_motors(command.data.steering.direction, command.data.steering.speed);
154                 break;
155         case COMMAND_TYPE_CAMERA:
156                 __camera(command.data.camera_position.camera_azimuth, command.data.camera_position.camera_elevation);
157                 break;
158         case COMMAND_TYPE_DRIVE_AND_CAMERA:
159                 __driving_motors(command.data.steering_and_camera.direction, command.data.steering_and_camera.speed);
160                 __camera(command.data.steering_and_camera.camera_azimuth, command.data.steering_and_camera.camera_elevation);
161                 break;
162         case COMMAND_TYPE_NONE:
163                 break;
164         default:
165                 _E("Unknown command type");
166                 break;
167         }
168 }
169
170 static void __user_name_received_cb(const char *name) {
171         lap_counter_set_user_name(name);
172 }
173
174 static void _initialize_config()
175 {
176         net_util_init();
177
178         config_init();
179
180         char *id = NULL;
181         char *name = NULL;
182         gboolean modified = false;
183         if (config_get_string(CONFIG_GRP_CAR, CONFIG_KEY_ID, &id) != 0) {
184                 char *uuid = g_uuid_string_random();
185                 config_set_string(CONFIG_GRP_CAR, CONFIG_KEY_ID, uuid);
186                 g_free(uuid);
187                 modified = true;
188         }
189         if (config_get_string(CONFIG_GRP_CAR, CONFIG_KEY_NAME, &id) != 0) {
190                 config_set_string(CONFIG_GRP_CAR, CONFIG_KEY_NAME, "Passerati");
191                 modified = true;
192         }
193         if (modified == true) {
194                 config_save();
195         }
196         free(id);
197         free(name);
198 }
199
200 static void _initialize_components(app_data *ad)
201 {
202         net_util_init();
203         _initialize_config();
204         cloud_communication_init();
205         message_manager_init();
206         controller_connection_manager_listen();
207         lap_counter_init();
208 }
209
210 static bool service_app_create(void *data)
211 {
212         int ret = 0;
213         app_data *ad = data;
214
215         _D("-----------------------=======================================================================-----------------------");
216         _D("-----------------------=======================================================================-----------------------");
217         _D("-----------------------============================== APP %s %s ==============================-----------------------", __DATE__, __TIME__);
218         _D("-----------------------=======================================================================-----------------------");
219         _D("-----------------------=======================================================================-----------------------");
220
221         /*
222          * if you want to use default configuration,
223          * Do not need to call resource_set_motor_driver_L298N_configuration(),
224          *
225         */
226 #if ENABLE_MOTOR
227         ret = resource_set_motor_driver_L298N_configuration(MOTOR_ID_1, 19, 16, 5);
228         if (ret) {
229                 _E("resource_set_motor_driver_L298N_configuration()");
230                 service_app_exit();
231         }
232         ret = resource_set_motor_driver_L298N_configuration(MOTOR_ID_2, 26, 20, 4);
233         if (ret) {
234                 _E("resource_set_motor_driver_L298N_configuration()");
235                 service_app_exit();
236         }
237 #endif
238
239         resource_lap_counter_init();
240
241         _initialize_components(ad);
242         cloud_communication_start(CLOUD_REQUESTS_FREQUENCY);
243
244         controller_connection_manager_set_command_received_cb(__command_received_cb);
245         controller_connection_manager_set_user_name_received_cb(__user_name_received_cb);
246
247         return true;
248 }
249
250 static void service_app_control(app_control_h app_control, void *data)
251 {
252
253 #if ENABLE_MOTOR
254         /* set speed 0, to reduce delay of initializing motor driver */
255         resource_set_motor_driver_L298N_speed(MOTOR_ID_1, 0);
256         resource_set_motor_driver_L298N_speed(MOTOR_ID_2, 0);
257         resource_set_servo_motor_value(0, 450);
258         resource_set_servo_motor_value(ELEVATION_SERVO_PIN, ELEVATION_MIN);
259         resource_set_servo_motor_value(AZIMUTH_SERVO_PIN, (AZIMUTH_MIN + AZIMUTH_MAX) / 2);
260 #endif
261
262         return;
263 }
264
265 static void service_app_terminate(void *data)
266 {
267         app_data *ad = data;
268
269         resource_set_servo_motor_value(0, STERING_SERVO_CENTER);
270         resource_set_servo_motor_value(ELEVATION_SERVO_PIN, ELEVATION_MIN);
271         resource_set_servo_motor_value(AZIMUTH_SERVO_PIN, (AZIMUTH_MIN + AZIMUTH_MAX) / 2);
272
273         resource_lap_counter_destroy();
274
275         if (ad->idle_h)
276                 g_source_remove(ad->idle_h);
277
278         lap_counter_shutdown();
279         controller_connection_manager_release();
280         message_manager_shutdown();
281
282         cloud_communication_stop();
283         cloud_communication_fini();
284         config_shutdown();
285         net_util_fini();
286
287         resource_close_all();
288         log_file_close();
289
290         _D("Bye ~");
291
292         return;
293 }
294
295 int main(int argc, char* argv[])
296 {
297         app_data *ad = NULL;
298         int ret = 0;
299         service_app_lifecycle_callback_s event_callback;
300         app_event_handler_h handlers[5] = {NULL, };
301
302         log_type_set(LOG_TYPE_DLOG);
303
304         ad = calloc(1, sizeof(app_data));
305         retv_if(!ad, -1);
306
307         event_callback.create = service_app_create;
308         event_callback.terminate = service_app_terminate;
309         event_callback.app_control = service_app_control;
310
311         service_app_add_event_handler(&handlers[APP_EVENT_LOW_BATTERY],
312                 APP_EVENT_LOW_BATTERY, service_app_low_battery, &ad);
313         service_app_add_event_handler(&handlers[APP_EVENT_LOW_MEMORY],
314                 APP_EVENT_LOW_MEMORY, service_app_low_memory, &ad);
315         service_app_add_event_handler(&handlers[APP_EVENT_LANGUAGE_CHANGED],
316                 APP_EVENT_LANGUAGE_CHANGED, service_app_lang_changed, &ad);
317         service_app_add_event_handler(&handlers[APP_EVENT_REGION_FORMAT_CHANGED],
318                 APP_EVENT_REGION_FORMAT_CHANGED, service_app_region_changed, &ad);
319
320         ret = service_app_main(argc, argv, &event_callback, ad);
321         if (ret)
322                 _E("failed to start app");
323
324         return 0;
325 }