/* * tio - a serial device I/O tool * * Copyright (c) 2014-2022 Martin Lund * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License * as published by the Free Software Foundation; either version 2 * of the License, or (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA * 02110-1301, USA. */ #define _GNU_SOURCE // For FNM_EXTMATCH #include #include #include #include #include #include #include #include "print.h" void delay(long ms) { struct timespec ts; if (ms <= 0) { return; } ts.tv_sec = ms / 1000; ts.tv_nsec = (ms % 1000) * 1000000; nanosleep(&ts, NULL); } int ctrl_key_code(unsigned char key) { if ((key >= 'a') && (key <= 'z')) { return key & ~0x60; } return -1; } bool regex_match(const char *string, const char *pattern) { regex_t regex; int status; if (regcomp(®ex, pattern, REG_EXTENDED | REG_NOSUB) != 0) { // No match return false; } status = regexec(®ex, string, (size_t) 0, NULL, 0); regfree(®ex); if (status != 0) { // No match return false; } // Match return true; } int read_poll(int fd, void *data, size_t len, int timeout) { struct pollfd fds; int ret = 0; fds.events = POLLIN; fds.fd = fd; /* Wait data available */ ret = poll(&fds, 1, timeout); if (ret < 0) { tio_error_print("%s", strerror(errno)); return ret; } else if (ret > 0) { if (fds.revents & POLLIN) { // Read ready data return read(fd, data, len); } } /* Timeout */ return ret; } // Function to calculate djb2 hash of string unsigned long djb2_hash(const unsigned char *str) { unsigned long hash = 5381; int c; while ((c = *str++)) { hash = ((hash << 5) + hash) + c; // hash * 33 + c } return hash; } // Function to encode a number to base62 char *base62_encode(unsigned long num) { const char base62_chars[] = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789"; char *output = (char *) malloc(5); // 4 characters + null terminator if (output == NULL) { tio_error_print("Memory allocation failed"); exit(EXIT_FAILURE); } for (int i = 0; i < 4; ++i) { output[i] = base62_chars[num % 62]; num /= 62; } output[4] = '\0'; return output; } // Function to return current time double get_current_time(void) { struct timespec current_time_ts; if (clock_gettime(CLOCK_REALTIME, ¤t_time_ts) == -1) { // Error return -1; } return current_time_ts.tv_sec + current_time_ts.tv_nsec / 1e9; } bool match_patterns(const char *string, const char *patterns) { char *pattern; char *patterns_copy; if ((string == NULL) || (patterns == NULL)) { return false; } patterns_copy = strdup(patterns); // Tokenize the patterns string using strtok pattern = strtok(patterns_copy, ","); while (pattern != NULL) { // Check if the string matches the current pattern #ifdef FNM_EXTMATCH if (fnmatch(pattern, string, FNM_EXTMATCH) == 0) #else if (fnmatch(pattern, string, 0) == 0) #endif { free(patterns_copy); return true; } // Move to the next pattern pattern = strtok(NULL, ","); } free(patterns_copy); return false; } // Function that forks subprocess, redirects its stdin and stdout to the // specified filedescriptor, and runs command. int execute_shell_command(int fd, const char *command) { pid_t pid; int status; // Fork a child process pid = fork(); if (pid == -1) { // Error occurred tio_error_print("fork() failed (%s)", strerror(errno)); exit(EXIT_FAILURE); } else if (pid == 0) { // Child process tio_printf("Executing shell command '%s'", command); // Redirect stdout and stderr to the file descriptor if (dup2(fd, STDOUT_FILENO) == -1 || dup2(fd, STDERR_FILENO) == -1) { tio_error_print("dup2() failed (%s)", strerror(errno)); exit(EXIT_FAILURE); } // Execute the shell command execl("/bin/sh", "sh", "-c", command, (char *)NULL); // If execlp() returns, it means an error occurred perror("execlp"); tio_error_print("execlp() failed (%s)", strerror(errno)); exit(EXIT_FAILURE); } else { // Parent process // Wait for the child process to finish waitpid(pid, &status, 0); if (WIFEXITED(status)) { tio_printf("Command exited with status %d", WEXITSTATUS(status)); return WEXITSTATUS(status); } else { tio_error_printf("Child process exited abnormally\n"); return -1; } } return 0; } void clear_line() { print("\r\033[K"); }