1 //-----------------------------------------------------------------------------
2 // Copyright (C) 2009 Michael Gernoth <michael at gernoth.net>
3 // Copyright (C) 2010 iZsh <izsh at fail0verflow.com>
5 // This code is licensed to you under the terms of the GNU GPL, version 2 or,
6 // at your option, any later version. See the LICENSE.txt file for the text of
8 //-----------------------------------------------------------------------------
10 //-----------------------------------------------------------------------------
17 #include <readline/readline.h>
18 #include <readline/history.h>
20 #include "proxmark3.h"
25 #include "cmdparser.h"
30 // a global mutex to prevent interlaced printing from different threads
31 pthread_mutex_t print_lock
;
33 static serial_port sp
;
34 static UsbCommand txcmd
;
35 volatile static bool txcmd_pending
= false;
37 void SendCommand(UsbCommand
*c
) {
39 printf("Sending %d bytes\n", sizeof(UsbCommand
));
43 PrintAndLog("Sending bytes to proxmark failed - offline");
47 The while-loop below causes hangups at times, when the pm3 unit is unresponsive
48 or disconnected. The main console thread is alive, but comm thread just spins here.
60 byte_t rx
[sizeof(UsbCommand
)];
63 static void *uart_receiver(void *targ
) {
64 struct receiver_arg
*arg
= (struct receiver_arg
*)targ
;
69 if (uart_receive(sp
, prx
, sizeof(UsbCommand
) - (prx
-rx
), &rxlen
)) {
71 if (prx
-rx
< sizeof(UsbCommand
)) {
75 UsbCommandReceived((UsbCommand
*)rx
);
80 if (!uart_send(sp
, (byte_t
*) &txcmd
, sizeof(UsbCommand
))) {
81 PrintAndLog("Sending bytes to proxmark failed");
83 txcmd_pending
= false;
92 void main_loop(char *script_cmds_file
, bool usb_present
) {
93 struct receiver_arg rarg
;
95 pthread_t reader_thread
;
99 pthread_create(&reader_thread
, NULL
, &uart_receiver
, &rarg
);
100 // cache Version information now:
104 FILE *script_file
= NULL
;
105 char script_cmd_buf
[256]; // iceman, needs lua script the same file_path_buffer as the rest
107 if (script_cmds_file
) {
108 script_file
= fopen(script_cmds_file
, "r");
110 printf("using 'scripting' commands file %s\n", script_cmds_file
);
114 read_history(".history");
118 // If there is a script file
121 if (!fgets(script_cmd_buf
, sizeof(script_cmd_buf
), script_file
)) {
126 nl
= strrchr(script_cmd_buf
, '\r');
129 nl
= strrchr(script_cmd_buf
, '\n');
132 if ((cmd
= (char*) malloc(strlen(script_cmd_buf
) + 1)) != NULL
) {
133 memset(cmd
, 0, strlen(script_cmd_buf
));
134 strcpy(cmd
, script_cmd_buf
);
141 cmd
= readline(PROXPROMPT
);
146 while(cmd
[strlen(cmd
) - 1] == ' ')
147 cmd
[strlen(cmd
) - 1] = 0x00;
149 if (cmd
[0] != 0x00) {
150 int ret
= CommandReceived(cmd
);
152 if (ret
== 99) { // exit or quit
163 write_history(".history");
167 pthread_join(reader_thread
, NULL
);
177 static void dumpAllHelp(int markdown
)
179 printf("\n%sProxmark3 command dump%s\n\n",markdown
?"# ":"",markdown
?"":"\n======================");
180 printf("Some commands are available only if a Proxmark is actually connected.%s\n",markdown
?" ":"");
181 printf("Check column \"offline\" for their availability.\n");
183 command_t
*cmds
= getTopLevelCommandTable();
184 dumpCommandsRecursive(cmds
, markdown
);
187 static char *my_executable_path
= NULL
;
188 static char *my_executable_directory
= NULL
;
190 const char *get_my_executable_path(void)
192 return my_executable_path
;
195 const char *get_my_executable_directory(void)
197 return my_executable_directory
;
200 static void set_my_executable_path(void)
202 int path_length
= wai_getExecutablePath(NULL
, 0, NULL
);
203 if (path_length
!= -1) {
204 my_executable_path
= (char*)malloc(path_length
+ 1);
205 int dirname_length
= 0;
206 if (wai_getExecutablePath(my_executable_path
, path_length
, &dirname_length
) != -1) {
207 my_executable_path
[path_length
] = '\0';
208 my_executable_directory
= (char *)malloc(dirname_length
+ 2);
209 strncpy(my_executable_directory
, my_executable_path
, dirname_length
+1);
210 my_executable_directory
[dirname_length
+1] = '\0';
216 int main(int argc
, char* argv
[]) {
220 printf("syntax: %s <port>\n\n",argv
[0]);
221 printf("\tLinux example:'%s /dev/ttyACM0'\n\n", argv
[0]);
222 printf("help: %s -h\n\n", argv
[0]);
223 printf("\tDump all interactive help at once\n");
224 printf("markdown: %s -m\n\n", argv
[0]);
225 printf("\tDump all interactive help at once in markdown syntax\n");
228 if (strcmp(argv
[1], "-h") == 0) {
229 printf("syntax: %s <port>\n\n",argv
[0]);
230 printf("\tLinux example:'%s /dev/ttyACM0'\n\n", argv
[0]);
234 if (strcmp(argv
[1], "-m") == 0) {
239 set_my_executable_path();
241 bool usb_present
= false;
242 char *script_cmds_file
= NULL
;
244 sp
= uart_open(argv
[1]);
245 if (sp
== INVALID_SERIAL_PORT
) {
246 printf("ERROR: invalid serial port\n");
249 } else if (sp
== CLAIMED_SERIAL_PORT
) {
250 printf("ERROR: serial port is claimed by another process\n");
258 // If the user passed the filename of the 'script' to execute, get it
259 if (argc
> 2 && argv
[2]) {
260 if (argv
[2][0] == 'f' && //buzzy, if a word 'flush' passed, flush the output after every log entry.
266 printf("Output will be flushed after every print.\n");
270 script_cmds_file
= argv
[2];
273 // create a mutex to avoid interlacing print commands from our different threads
274 pthread_mutex_init(&print_lock
, NULL
);
278 InitGraphics(argc
, argv
, script_cmds_file
, usb_present
);
281 char* display
= getenv("DISPLAY");
283 if (display
&& strlen(display
) > 1)
285 InitGraphics(argc
, argv
, script_cmds_file
, usb_present
);
290 main_loop(script_cmds_file
, usb_present
);
294 main_loop(script_cmds_file
, usb_present
);
303 pthread_mutex_destroy(&print_lock
);