1 /**************************************************************************
5 * Copyright 2015 Samsung Electronics co., Ltd. All Rights Reserved.
7 * Contact: Eunchul Kim <chulspro.kim@samsung.com>,
8 * JinYoung Jeon <jy0.jeon@samsung.com>,
9 * Taeheon Kim <th908.kim@samsung.com>,
10 * YoungJun Cho <yj44.cho@samsung.com>,
11 * SooChan Lim <sc1.lim@samsung.com>,
12 * Boram Park <boram1288.park@samsung.com>
14 * Permission is hereby granted, free of charge, to any person obtaining a
15 * copy of this software and associated documentation files (the
16 * "Software"), to deal in the Software without restriction, including
17 * without limitation the rights to use, copy, modify, merge, publish,
18 * distribute, sub license, and/or sell copies of the Software, and to
19 * permit persons to whom the Software is furnished to do so, subject to
20 * the following conditions:
22 * The above copyright notice and this permission notice (including the
23 * next paragraph) shall be included in all copies or substantial portions
26 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
27 * OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
28 * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT.
29 * IN NO EVENT SHALL PRECISION INSIGHT AND/OR ITS SUPPLIERS BE LIABLE FOR
30 * ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
31 * TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
32 * SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
34 **************************************************************************/
36 #include "tdm_private.h"
38 #define TDM_DBG_SERVER_ARGS_MAX 32
41 static void _tdm_monitor_server_usage(char *app_name, char *reply, int *len);
44 _tdm_monitor_server_query(unsigned int pid, char *cwd, int argc, char *argv[], char *reply, int *len, tdm_display *dpy)
46 tdm_helper_get_display_information(dpy, reply, len);
50 _tdm_monitor_server_protocol_trace(unsigned int pid, char *cwd, int argc, char *argv[], char *reply, int *len, tdm_display *dpy)
55 _tdm_monitor_server_usage(argv[0], reply, len);
59 enable = atoi(argv[2]);
61 tdm_event_loop_trace_enable(dpy, enable);
63 TDM_SNPRINTF(reply, len, "protocol trace: '%s'\n", (enable) ? "enabled" : "disabled");
67 _tdm_monitor_server_dpms(unsigned int pid, char *cwd, int argc, char *argv[], char *reply, int *len, tdm_display *dpy)
70 int output_idx, set, dpms_value = 0;
71 tdm_output_dpms dpms = TDM_OUTPUT_DPMS_ON;
77 _tdm_monitor_server_usage(argv[0], reply, len);
81 set = strtol(argv[2], &end, 10);
82 if (!(set == 0 || set ==1)) {
83 _tdm_monitor_server_usage(argv[0], reply, len);
89 output_idx = strtol(arg, &end, 10);
91 TDM_SNPRINTF(reply, len, "failed: no onoff value\n");
96 dpms_value = strtol(arg, &end, 10);
98 output = tdm_display_get_output(dpy, output_idx, &ret);
99 TDM_DBG_RETURN_IF_FAIL(ret == TDM_ERROR_NONE && output != NULL);
101 ret = tdm_output_set_dpms(output, dpms_value);
102 TDM_DBG_RETURN_IF_FAIL(ret == TDM_ERROR_NONE);
104 TDM_SNPRINTF(reply, len, "done: DPMS set %s\n", tdm_dpms_str(dpms_value));
106 output_idx = strtol(argv[3], &end, 10);
108 output = tdm_display_get_output(dpy, output_idx, &ret);
109 TDM_DBG_RETURN_IF_FAIL(ret == TDM_ERROR_NONE && output != NULL);
111 ret = tdm_output_get_dpms(output, &dpms);
112 TDM_DBG_RETURN_IF_FAIL(ret == TDM_ERROR_NONE);
114 TDM_SNPRINTF(reply, len, "done: DPMS get %s\n", tdm_dpms_str(dpms));
119 _tdm_monitor_server_ttrace(unsigned int pid, char *cwd, int argc, char *argv[], char *reply, int *len, tdm_display *dpy)
126 _tdm_monitor_server_usage(argv[0], reply, len);
131 output_id = strtol(argv[3], &end, 10);
133 ret = tdm_display_enable_ttrace(dpy, argv[2], output_id, reply, len);
134 TDM_DBG_RETURN_IF_FAIL(ret == TDM_ERROR_NONE);
138 _tdm_monitor_server_debug(unsigned int pid, char *cwd, int argc, char *argv[], char *reply, int *len, tdm_display *dpy)
145 _tdm_monitor_server_usage(argv[0], reply, len);
150 level = strtol(arg, &end, 10);
152 tdm_log_set_debug_level(level);
153 TDM_SNPRINTF(reply, len, "debug level: %d\n", level);
158 tdm_display_enable_debug_module((const char *)arg);
160 TDM_SNPRINTF(reply, len, "debugging... '%s'\n", arg);
165 _tdm_monitor_server_log_path(unsigned int pid, char *cwd, int argc, char *argv[], char *reply, int *len, tdm_display *dpy)
167 static int old_stdout = -1;
168 char fd_name[TDM_PATH_LEN];
172 _tdm_monitor_server_usage(argv[0], reply, len);
176 if (old_stdout == -1)
177 old_stdout = dup(STDOUT_FILENO);
180 TDM_DBG_RETURN_IF_FAIL(path != NULL);
182 tdm_log_enable_dlog(0);
184 if (!strncmp(path, "dlog", 4)) {
185 tdm_log_enable_dlog(1);
187 } else if (!strncmp(path, "console", 7))
188 snprintf(fd_name, TDM_PATH_LEN, "/proc/%d/fd/1", pid);
191 snprintf(fd_name, TDM_PATH_LEN, "%s", path);
194 snprintf(fd_name, TDM_PATH_LEN, "%s/%s", cwd, path);
196 snprintf(fd_name, TDM_PATH_LEN, "%s", path);
198 tdm_log_enable_color(0);
201 tdm_log_set_path((const char*)fd_name);
204 TDM_SNPRINTF(reply, len, "log path: '%s'\n", path);
208 _tdm_monitor_server_choose_commit_per_vblank_mode(unsigned int pid, char *cwd, int argc, char *argv[], char *reply, int *len, tdm_display *dpy)
216 /* TODO: do we have to provide an ability to choose commit_per_vblank mode outside? */
219 _tdm_monitor_server_usage(argv[0], reply, len);
224 output_idx = strtol(arg, &end, 10);
227 mode = strtol(arg, &end, 10);
229 err = tdm_output_choose_commit_per_vblank_mode(tdm_display_get_output(dpy, output_idx, NULL), mode);
230 if (err != TDM_ERROR_NONE) {
231 TDM_SNPRINTF(reply, len, "an error: output_idx or mode is wrong\n");
235 TDM_SNPRINTF(reply, len, "the mode's been set\n");
239 _tdm_monitor_server_fps(unsigned int pid, char *cwd, int argc, char *argv[], char *reply, int *len, tdm_display *dpy)
246 _tdm_monitor_server_usage(argv[0], reply, len);
251 enable = strtol(arg, &end, 10);
253 tdm_display_enable_fps(dpy, enable);
255 TDM_SNPRINTF(reply, len, "%s to print fps\n", (enable) ? "enable" : "disable");
259 _tdm_monitor_server_vblank_list(unsigned int pid, char *cwd, int argc, char *argv[],
260 char *reply, int *len, tdm_display *dpy)
262 tdm_vblank_get_vblank_list_information(dpy, reply, len);
266 _tdm_monitor_server_vblank_fps(unsigned int pid, char *cwd, int argc, char *argv[],
267 char *reply, int *len, tdm_display *dpy)
269 unsigned int target_pid, fps;
272 char name[TDM_NAME_LEN];
276 _tdm_monitor_server_usage(argv[0], reply, len);
281 target_pid = strtol(arg, &end, 10);
285 end = strtostr(name, TDM_NAME_LEN, arg, TDM_DELIM);
287 strncpy(name, TDM_VBLANK_DEFAULT_NAME, TDM_NAME_LEN - 1);
288 name[TDM_NAME_LEN - 1] = '\0';
292 TDM_SNPRINTF(reply, len, "failed: no fps value\n");
297 fps = strtol(arg, &end, 10);
299 ret = tdm_vblank_set_client_vblank_fps(target_pid, name, fps);
300 if (ret != TDM_ERROR_NONE) {
301 TDM_SNPRINTF(reply, len, "can't set '%u' fps to '%s' client vblank(PID:%u)\n", fps, name, target_pid);
305 TDM_SNPRINTF(reply, len, "success: '%u' fps for '%s' client vblank(PID:%u)\n", fps, name, target_pid);
309 _tdm_monitor_server_global_fps(unsigned int pid, char *cwd, int argc, char *argv[],
310 char *reply, int *len, tdm_display *dpy)
318 _tdm_monitor_server_usage(argv[0], reply, len);
323 fps = strtol(arg, &end, 10);
326 ret = tdm_vblank_enable_global_fps(1, fps);
328 ret = tdm_vblank_enable_global_fps(0, 0);
330 if (ret != TDM_ERROR_NONE) {
331 TDM_SNPRINTF(reply, len, "can't set '%u' fps to global fps\n", fps);
335 TDM_SNPRINTF(reply, len, "success: '%u' global fps\n", fps);
339 _tdm_monitor_server_prop(unsigned int pid, char *cwd, int argc, char *argv[], char *reply, int *len, tdm_display *dpy)
342 tdm_output *layer = NULL;
343 int output_idx, layer_idx = -1;
344 int cnt, i, done = 0;
346 char temp[TDM_PATH_LEN];
351 const tdm_prop *props;
354 _tdm_monitor_server_usage(argv[0], reply, len);
358 snprintf(temp, TDM_PATH_LEN, "%s", argv[2]);
361 output_idx = strtol(arg, &end, 10);
364 layer_idx = strtol(arg, &end, 10);
368 TDM_SNPRINTF(reply, len, "failed: no prop_name\n");
373 prop_name = strtok_r(arg, ",", &end);
375 TDM_SNPRINTF(reply, len, "failed: get prop_name by strtok_r\n");
380 TDM_SNPRINTF(reply, len, "failed: no value\n");
384 arg = strtok_r(NULL, TDM_DELIM, &end);
386 TDM_SNPRINTF(reply, len, "failed: get arg by strtok_r\n");
389 value.u32 = strtol(arg, &end, 10);
391 output = tdm_display_get_output(dpy, output_idx, &ret);
392 TDM_DBG_RETURN_IF_FAIL(ret == TDM_ERROR_NONE && output != NULL);
394 if (layer_idx != -1) {
395 layer = tdm_output_get_layer(output, layer_idx, &ret);
396 TDM_DBG_RETURN_IF_FAIL(ret == TDM_ERROR_NONE && layer != NULL);
400 ret = tdm_layer_get_available_properties(layer, &props, &cnt);
401 TDM_DBG_RETURN_IF_FAIL(ret == TDM_ERROR_NONE);
403 for (i = 0; i < cnt; i++) {
404 if (!strncmp(props[i].name, prop_name, TDM_NAME_LEN)) {
405 ret = tdm_layer_set_property(layer, props[i].id, value);
406 TDM_DBG_RETURN_IF_FAIL(ret == TDM_ERROR_NONE);
412 ret = tdm_output_get_available_properties(output, &props, &cnt);
413 TDM_DBG_RETURN_IF_FAIL(ret == TDM_ERROR_NONE);
415 for (i = 0; i < cnt; i++) {
416 if (!strncmp(props[i].name, prop_name, TDM_NAME_LEN)) {
417 ret = tdm_output_set_property(output, props[i].id, value);
418 TDM_DBG_RETURN_IF_FAIL(ret == TDM_ERROR_NONE);
426 TDM_SNPRINTF(reply, len, "done: %s:%d \n", prop_name, value.u32);
428 TDM_SNPRINTF(reply, len, "no '%s' propperty \n", prop_name);
432 _tdm_monitor_server_dump(unsigned int pid, char *cwd, int argc, char *argv[], char *reply, int *len, tdm_display *dpy)
435 _tdm_monitor_server_usage(argv[0], reply, len);
439 tdm_display_enable_dump(dpy, (const char*)argv[2], reply, len);
443 _tdm_monitor_server_punch(unsigned int pid, char *cwd, int argc, char *argv[], char *reply, int *len, tdm_display *dpy)
446 unsigned int output_id, layer_id;
449 tbm_surface_h buffer;
453 output_id = strtol(arg, &end, 10);
454 output = tdm_display_get_output(dpy, output_id, NULL);
456 TDM_SNPRINTF(reply, len, "not found output\n");
460 TDM_SNPRINTF(reply, len, "not found ',<layer_idx>'\n");
465 layer_id = strtol(arg, &end, 10);
466 layer = tdm_output_get_layer(output, layer_id, NULL);
468 TDM_SNPRINTF(reply, len, "not found layer\n");
472 buffer = tdm_layer_get_displaying_buffer(layer, NULL);
474 TDM_SNPRINTF(reply, len, "not found buffer\n");
482 pos.w = strtol(arg, &end, 10);
483 TDM_EXIT_IF_FAIL(*end == 'x');
485 pos.h = strtol(arg, &end, 10);
488 pos.x = strtol(arg, &end, 10);
489 TDM_EXIT_IF_FAIL(*end == '+');
491 pos.y = strtol(arg, &end, 10);
494 tdm_helper_clear_buffer_pos(buffer, &pos);
496 tdm_helper_clear_buffer(buffer);
501 void (*func)(unsigned int pid, char *cwd, int argc, char *argv[], char *reply, int *len, tdm_display *dpy);
507 "info", _tdm_monitor_server_query,
508 "show tdm output, layer information", NULL, NULL
511 "trace", _tdm_monitor_server_protocol_trace,
512 "enable/disable the wl protocol",
517 "dpms", _tdm_monitor_server_dpms,
518 "set/get output dpms",
519 "<set/get> <output_idx>:<dpms>",
520 "0 0 (get) or 1 0:3 (set off) or 1 0:0 (set on)"
523 "ttrace", _tdm_monitor_server_ttrace,
524 "enable/disable ttrace (module: none,vsync,client_vblank,server_vblank,vblank,layer,pp,capture,all",
525 "<module>[@<output_idx>]",
529 "debug", _tdm_monitor_server_debug,
530 "set the debug level and modules(module: none, mutex, buffer, thread, vblank)",
531 "<level>[@<module1>[,<module2>]]",
535 "log_path", _tdm_monitor_server_log_path,
536 "set the log path (console,dlog,filepath)",
541 "choose_commit_per_vblank_mode", _tdm_monitor_server_choose_commit_per_vblank_mode,
542 "choose a commit_per_vblank mode for the output",
543 "<output_idx> <mode>",
547 "fps", _tdm_monitor_server_fps,
548 "enable/disable to print fps",
553 "vblank_list", _tdm_monitor_server_vblank_list,
554 "print the client vblank list",
559 "vblank_fps", _tdm_monitor_server_vblank_fps,
560 "set the client vblank fps for the given process ID and client vblank name",
561 "<pid>[,<vblank_name>]@<fps>",
565 "global_fps", _tdm_monitor_server_global_fps,
566 "set the global vblank fps for the entire system",
571 "prop", _tdm_monitor_server_prop,
572 "set the property of a output or a layer",
573 "<output_idx>[,<layer_idx>]:<prop_name>,<value>",
577 "dump", _tdm_monitor_server_dump,
578 "dump buffers (type: none, window, layer, pp, capture, current)\n"
579 "\t\t window, layer, pp, capture - start to dump buffers of window, layer, pp, capture\n"
580 "\t\t none - stop to dump buffers\n"
581 "\t\t current - dump the current buffer of all layers",
582 "<object_type1>[,<object_type2>[,...]][@<directory_path>]",
586 "punch", _tdm_monitor_server_punch,
588 "<output_idx>,<layer_idx>[:<w>x<h>[+<x>+<y>]]",
594 _tdm_monitor_server_usage(char *app_name, char *reply, int *len)
596 int opt_size = sizeof(option_proc) / sizeof(option_proc[0]);
599 TDM_SNPRINTF(reply, len, "usage: %s \n\n", app_name);
601 for (i = 0; i < opt_size; i++) {
602 TDM_SNPRINTF(reply, len, "\t-%s\t%s\n", option_proc[i].opt, option_proc[i].desc);
603 if (option_proc[i].arg)
604 TDM_SNPRINTF(reply, len, "\t\t %s\n", option_proc[i].arg);
605 if (option_proc[i].ex)
606 TDM_SNPRINTF(reply, len, "\t\t ex) %s\n", option_proc[i].ex);
607 TDM_SNPRINTF(reply, len, "\n");
612 _tdm_monitor_server_command(unsigned int pid, char *cwd, tdm_display *dpy, int argc, char *argv[], char *reply, int *len)
614 int opt_size = sizeof(option_proc) / sizeof(option_proc[0]);
618 _tdm_monitor_server_usage(argv[0], reply, len);
622 for (i = 0; i < opt_size; i++) {
623 if (argv[1][0] == '-' && !strncmp(argv[1] + 1, option_proc[i].opt, 32)) {
624 if (option_proc[i].func) {
625 option_proc[i].func(pid, cwd, argc, argv, reply, len, dpy);
628 TDM_SNPRINTF(reply, len, "'%s' not implemented.\n", argv[1]);
634 _tdm_monitor_server_usage(argv[0], reply, len);
640 tdm_monitor_server_command(tdm_display *dpy, const char *options, char *reply, int *len)
645 char *argv[TDM_DBG_SERVER_ARGS_MAX] = {0,};
648 char *end = NULL, *e;
650 snprintf(temp, sizeof(temp), "%s", options);
652 arg = strtok_r(temp, " ", &end);
654 TDM_SNPRINTF(reply, len, "no pid for tdm-monitor");
657 pid = strtol(arg, &e, 10);
659 arg = strtok_r(NULL, " ", &end);
661 TDM_SNPRINTF(reply, len, "no cwd for tdm-monitor");
664 snprintf(cwd, sizeof(cwd), "%s", arg);
666 TDM_DBG("pid(%d) cwd(%s)", pid, cwd);
668 argv[argc] = strtok_r(NULL, " ", &end);
671 if (argc == TDM_DBG_SERVER_ARGS_MAX) {
672 TDM_SNPRINTF(reply, len, "too many arguments for tdm-monitor");
675 argv[argc] = strtok_r(NULL, " ", &end);
678 _tdm_monitor_server_command(pid, cwd, dpy, argc, argv, reply, len);