Bert Vermeulen | 2be182e | 2013-11-17 13:15:38 +0100 | [diff] [blame] | 1 | /* |
| 2 | * This file is part of the sigrok-cli project. |
| 3 | * |
| 4 | * Copyright (C) 2013 Bert Vermeulen <bert@biot.com> |
| 5 | * |
| 6 | * This program is free software: you can redistribute it and/or modify |
| 7 | * it under the terms of the GNU General Public License as published by |
| 8 | * the Free Software Foundation, either version 3 of the License, or |
| 9 | * (at your option) any later version. |
| 10 | * |
| 11 | * This program is distributed in the hope that it will be useful, |
| 12 | * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| 13 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
| 14 | * GNU General Public License for more details. |
| 15 | * |
| 16 | * You should have received a copy of the GNU General Public License |
| 17 | * along with this program. If not, see <http://www.gnu.org/licenses/>. |
| 18 | */ |
| 19 | |
Uwe Hermann | 20fb52e | 2013-11-19 11:48:06 +0100 | [diff] [blame] | 20 | #include "sigrok-cli.h" |
Bert Vermeulen | 2be182e | 2013-11-17 13:15:38 +0100 | [diff] [blame] | 21 | #include "config.h" |
Bert Vermeulen | a8b4041 | 2014-08-10 16:58:36 +0200 | [diff] [blame] | 22 | #include <sys/types.h> |
Dan Horák | 8d52f78 | 2014-01-25 14:15:24 +0100 | [diff] [blame] | 23 | #include <sys/stat.h> |
Bert Vermeulen | a8b4041 | 2014-08-10 16:58:36 +0200 | [diff] [blame] | 24 | #include <fcntl.h> |
| 25 | #include <unistd.h> |
Dan Horák | 8d52f78 | 2014-01-25 14:15:24 +0100 | [diff] [blame] | 26 | #include <errno.h> |
| 27 | #include <stdlib.h> |
| 28 | #include <string.h> |
Bert Vermeulen | 2be182e | 2013-11-17 13:15:38 +0100 | [diff] [blame] | 29 | #include <glib.h> |
Bert Vermeulen | 2be182e | 2013-11-17 13:15:38 +0100 | [diff] [blame] | 30 | |
Bert Vermeulen | a8b4041 | 2014-08-10 16:58:36 +0200 | [diff] [blame] | 31 | #define BUFSIZE 16384 |
| 32 | |
| 33 | static void load_input_file_module(void) |
Bert Vermeulen | 2be182e | 2013-11-17 13:15:38 +0100 | [diff] [blame] | 34 | { |
Bert Vermeulen | 4bf77ec | 2014-07-17 22:49:20 +0200 | [diff] [blame] | 35 | struct sr_session *session; |
Bert Vermeulen | a8b4041 | 2014-08-10 16:58:36 +0200 | [diff] [blame] | 36 | const struct sr_input *in; |
| 37 | const struct sr_input_module *imod; |
| 38 | const struct sr_option **options; |
| 39 | struct sr_dev_inst *sdi; |
| 40 | GHashTable *mod_args, *mod_opts; |
| 41 | GString *buf; |
Uwe Hermann | c5136a5 | 2014-09-19 12:39:33 +0200 | [diff] [blame] | 42 | int fd = 0; |
Bert Vermeulen | a8b4041 | 2014-08-10 16:58:36 +0200 | [diff] [blame] | 43 | ssize_t len; |
| 44 | char *mod_id; |
Bert Vermeulen | 2be182e | 2013-11-17 13:15:38 +0100 | [diff] [blame] | 45 | |
Bert Vermeulen | a8b4041 | 2014-08-10 16:58:36 +0200 | [diff] [blame] | 46 | if (!sr_input_list()) |
| 47 | g_critical("No supported input formats available."); |
| 48 | |
| 49 | mod_id = NULL; |
| 50 | mod_args = NULL; |
Bert Vermeulen | 2be182e | 2013-11-17 13:15:38 +0100 | [diff] [blame] | 51 | if (opt_input_format) { |
Bert Vermeulen | a8b4041 | 2014-08-10 16:58:36 +0200 | [diff] [blame] | 52 | mod_args = parse_generic_arg(opt_input_format, TRUE); |
| 53 | mod_id = g_hash_table_lookup(mod_args, "sigrok_key"); |
Bert Vermeulen | 2be182e | 2013-11-17 13:15:38 +0100 | [diff] [blame] | 54 | } |
| 55 | |
Bert Vermeulen | a8b4041 | 2014-08-10 16:58:36 +0200 | [diff] [blame] | 56 | buf = g_string_sized_new(BUFSIZE); |
| 57 | if (mod_id) { |
| 58 | /* User specified an input module to use. */ |
| 59 | if (!(imod = sr_input_find(mod_id))) |
| 60 | g_critical("Error: unknown input module '%s'.", mod_id); |
| 61 | g_hash_table_remove(mod_args, "sigrok_key"); |
| 62 | if ((options = sr_input_options_get(imod))) { |
| 63 | mod_opts = generic_arg_to_opt(options, mod_args); |
| 64 | sr_output_options_free(options); |
| 65 | } else |
| 66 | mod_opts = NULL; |
| 67 | if (!(in = sr_input_new(imod, mod_opts))) |
| 68 | g_critical("Error: failed to initialize input module."); |
| 69 | if (mod_opts) |
| 70 | g_hash_table_destroy(mod_opts); |
| 71 | if (mod_args) |
| 72 | g_hash_table_destroy(mod_args); |
| 73 | if ((fd = open(opt_input_file, O_RDONLY)) == -1) |
| 74 | g_critical("Failed to load %s: %s.", opt_input_file, |
| 75 | strerror(errno)); |
| 76 | } else { |
Bert Vermeulen | 47f9741 | 2014-08-20 01:15:36 +0200 | [diff] [blame] | 77 | if (strcmp(opt_input_file, "-")) { |
| 78 | /* |
| 79 | * An actual filename: let the input modules try to |
| 80 | * identify the file. |
| 81 | */ |
Bert Vermeulen | b66260b | 2014-08-28 01:50:49 +0200 | [diff] [blame] | 82 | if (sr_input_scan_file(opt_input_file, &in) == SR_OK) { |
| 83 | /* That worked, reopen the file for reading. */ |
| 84 | fd = open(opt_input_file, O_RDONLY); |
| 85 | } |
Bert Vermeulen | 47f9741 | 2014-08-20 01:15:36 +0200 | [diff] [blame] | 86 | } else { |
| 87 | /* |
| 88 | * Taking input from a pipe: let the input modules try |
| 89 | * to identify the stream content. |
| 90 | */ |
| 91 | if (!strcmp(opt_input_file, "-")) { |
| 92 | /* stdin */ |
| 93 | fd = 0; |
| 94 | } else { |
| 95 | if ((fd = open(opt_input_file, O_RDONLY)) == -1) |
| 96 | g_critical("Failed to load %s: %s.", opt_input_file, |
| 97 | strerror(errno)); |
| 98 | } |
| 99 | if ((len = read(fd, buf->str, BUFSIZE)) < 1) |
| 100 | g_critical("Failed to read %s: %s.", opt_input_file, |
| 101 | strerror(errno)); |
| 102 | buf->len = len; |
Bert Vermeulen | b66260b | 2014-08-28 01:50:49 +0200 | [diff] [blame] | 103 | sr_input_scan_buffer(buf, &in); |
Bert Vermeulen | 47f9741 | 2014-08-20 01:15:36 +0200 | [diff] [blame] | 104 | } |
| 105 | if (!in) |
Bert Vermeulen | a8b4041 | 2014-08-10 16:58:36 +0200 | [diff] [blame] | 106 | g_critical("Error: no input module found for this file."); |
Bert Vermeulen | 2be182e | 2013-11-17 13:15:38 +0100 | [diff] [blame] | 107 | } |
Bert Vermeulen | a8b4041 | 2014-08-10 16:58:36 +0200 | [diff] [blame] | 108 | sdi = sr_input_dev_inst_get(in); |
Bert Vermeulen | 2be182e | 2013-11-17 13:15:38 +0100 | [diff] [blame] | 109 | |
Bert Vermeulen | a8b4041 | 2014-08-10 16:58:36 +0200 | [diff] [blame] | 110 | if (select_channels(sdi) != SR_OK) |
Bert Vermeulen | 2be182e | 2013-11-17 13:15:38 +0100 | [diff] [blame] | 111 | return; |
| 112 | |
Bert Vermeulen | 4bf77ec | 2014-07-17 22:49:20 +0200 | [diff] [blame] | 113 | sr_session_new(&session); |
| 114 | sr_session_datafeed_callback_add(session, &datafeed_in, NULL); |
Bert Vermeulen | a8b4041 | 2014-08-10 16:58:36 +0200 | [diff] [blame] | 115 | if (sr_session_dev_add(session, sdi) != SR_OK) { |
Bert Vermeulen | 2be182e | 2013-11-17 13:15:38 +0100 | [diff] [blame] | 116 | g_critical("Failed to use device."); |
Bert Vermeulen | 4bf77ec | 2014-07-17 22:49:20 +0200 | [diff] [blame] | 117 | sr_session_destroy(session); |
Bert Vermeulen | 2be182e | 2013-11-17 13:15:38 +0100 | [diff] [blame] | 118 | return; |
| 119 | } |
| 120 | |
Bert Vermeulen | a8b4041 | 2014-08-10 16:58:36 +0200 | [diff] [blame] | 121 | while(TRUE) { |
| 122 | g_string_truncate(buf, 0); |
| 123 | len = read(fd, buf->str, BUFSIZE); |
| 124 | if (len < 0) |
| 125 | g_critical("Read failed: %s", strerror(errno)); |
| 126 | buf->len = len; |
| 127 | if (sr_input_send(in, buf) != SR_OK) |
| 128 | break; |
| 129 | if (len < BUFSIZE) |
| 130 | break; |
| 131 | } |
| 132 | sr_input_free(in); |
| 133 | g_string_free(buf, TRUE); |
Bert Vermeulen | 2be182e | 2013-11-17 13:15:38 +0100 | [diff] [blame] | 134 | |
Bert Vermeulen | 4bf77ec | 2014-07-17 22:49:20 +0200 | [diff] [blame] | 135 | sr_session_destroy(session); |
Bert Vermeulen | 2be182e | 2013-11-17 13:15:38 +0100 | [diff] [blame] | 136 | |
Bert Vermeulen | 2be182e | 2013-11-17 13:15:38 +0100 | [diff] [blame] | 137 | } |
| 138 | |
| 139 | void load_input_file(void) |
| 140 | { |
Bert Vermeulen | 4bf77ec | 2014-07-17 22:49:20 +0200 | [diff] [blame] | 141 | struct sr_session *session; |
Daniel Elstner | e1ec80f | 2014-02-21 00:34:12 +0100 | [diff] [blame] | 142 | struct sr_dev_inst *sdi; |
Bert Vermeulen | a8b4041 | 2014-08-10 16:58:36 +0200 | [diff] [blame] | 143 | GSList *devices; |
Daniel Elstner | e1ec80f | 2014-02-21 00:34:12 +0100 | [diff] [blame] | 144 | int ret; |
Bert Vermeulen | 2be182e | 2013-11-17 13:15:38 +0100 | [diff] [blame] | 145 | |
Bert Vermeulen | 11c747b | 2014-08-29 23:02:19 +0200 | [diff] [blame] | 146 | if (strcmp(opt_input_file, "-") && sr_session_load(opt_input_file, &session) == SR_OK) { |
Bert Vermeulen | 2be182e | 2013-11-17 13:15:38 +0100 | [diff] [blame] | 147 | /* sigrok session file */ |
Bert Vermeulen | 4bf77ec | 2014-07-17 22:49:20 +0200 | [diff] [blame] | 148 | ret = sr_session_dev_list(session, &devices); |
| 149 | if (ret != SR_OK || !devices->data) { |
Daniel Elstner | e1ec80f | 2014-02-21 00:34:12 +0100 | [diff] [blame] | 150 | g_critical("Failed to access session device."); |
Bert Vermeulen | 4bf77ec | 2014-07-17 22:49:20 +0200 | [diff] [blame] | 151 | sr_session_destroy(session); |
Daniel Elstner | e1ec80f | 2014-02-21 00:34:12 +0100 | [diff] [blame] | 152 | return; |
| 153 | } |
Bert Vermeulen | 4bf77ec | 2014-07-17 22:49:20 +0200 | [diff] [blame] | 154 | sdi = devices->data; |
Uwe Hermann | 029d73f | 2014-03-24 22:32:47 +0100 | [diff] [blame] | 155 | if (select_channels(sdi) != SR_OK) { |
Bert Vermeulen | 4bf77ec | 2014-07-17 22:49:20 +0200 | [diff] [blame] | 156 | sr_session_destroy(session); |
Daniel Elstner | e1ec80f | 2014-02-21 00:34:12 +0100 | [diff] [blame] | 157 | return; |
| 158 | } |
Bert Vermeulen | 4bf77ec | 2014-07-17 22:49:20 +0200 | [diff] [blame] | 159 | sr_session_datafeed_callback_add(session, datafeed_in, NULL); |
| 160 | sr_session_start(session); |
| 161 | sr_session_run(session); |
| 162 | sr_session_stop(session); |
Bert Vermeulen | 11c747b | 2014-08-29 23:02:19 +0200 | [diff] [blame] | 163 | } else { |
Bert Vermeulen | 2be182e | 2013-11-17 13:15:38 +0100 | [diff] [blame] | 164 | /* fall back on input modules */ |
Bert Vermeulen | a8b4041 | 2014-08-10 16:58:36 +0200 | [diff] [blame] | 165 | load_input_file_module(); |
Bert Vermeulen | 2be182e | 2013-11-17 13:15:38 +0100 | [diff] [blame] | 166 | } |
| 167 | } |