| /* Copyright (C) 2021-2024 Free Software Foundation, Inc. |
| Contributed by Oracle. |
| |
| This file is part of GNU Binutils. |
| |
| 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 3, 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, 51 Franklin Street - Fifth Floor, Boston, |
| MA 02110-1301, USA. */ |
| |
| #include "config.h" |
| #include <string.h> |
| #include <unistd.h> |
| #include <dirent.h> |
| |
| #include "DbeSession.h" |
| #include "Command.h" |
| #include "Application.h" |
| #include "MemorySpace.h" |
| #include "i18n.h" |
| |
| #define MAXARGS 20 |
| |
| static const char *LIBNAME = "../lib/analyzer/lib/machinemodels"; |
| |
| char * |
| DbeSession::find_mach_model (char *name) |
| { |
| // Read current working directory to see if it's there |
| if (name[0] == '/') |
| { |
| // Absolute path given |
| char *path = dbe_sprintf (NTXT ("%s.ermm"), name); |
| if (access (path, R_OK | F_OK) == 0) |
| return path; |
| free (path); |
| // Don't try anywhere else |
| return NULL; |
| } |
| |
| char *path = dbe_sprintf (NTXT ("./%s.ermm"), name); |
| if (access (path, R_OK | F_OK) == 0) |
| return path; |
| free (path); |
| |
| |
| // Read the user's home directory to see if it's there |
| char *home = getenv (NTXT ("HOME")); |
| if (home != NULL) |
| { |
| path = dbe_sprintf (NTXT ("%s/%s.ermm"), home, name); |
| if (access (path, R_OK | F_OK) == 0) |
| return path; |
| free (path); |
| } |
| if (strchr (name, (int) '/') != NULL) |
| // name has a slash; don't look in system installation directory |
| return NULL; |
| |
| // Read system installation directory to see if it's there |
| path = dbe_sprintf ("%s/%s/%s.ermm", theApplication->get_run_dir (), |
| LIBNAME, name); |
| if (access (path, R_OK | F_OK) == 0) |
| return path; |
| free (path); |
| return NULL; |
| } |
| |
| // Handle the definitions from a machinemodel file |
| // Return value is NULL if it was OK, or an error message if not |
| char * |
| DbeSession::load_mach_model (char *_name) |
| { |
| CmdType cmd_type; |
| int arg_count, cparam; |
| char *cmd, *end_cmd; |
| char *arglist[MAXARGS]; |
| char *ret = NULL; |
| char *path = NULL; |
| FILE *fptr = NULL; |
| |
| // Does name have .ermm suffix? If so, strip it away |
| char *name = dbe_strdup (_name); |
| size_t len = strlen (name); |
| if (len > 5 && strcmp (name + len - 5, ".ermm") == 0) |
| name[len - 5] = 0; |
| |
| if ((mach_model_loaded != NULL) && (strcmp (name, mach_model_loaded) == 0)) |
| { |
| ret = dbe_sprintf (GTXT ("Machine model %s is already loaded\n"), name); |
| free (name); |
| return ret; |
| } |
| else if (mach_model_loaded == NULL && len == 0) |
| { |
| ret = dbe_sprintf (GTXT ("No Machine model is loaded\n")); |
| free (name); |
| return ret; |
| } |
| |
| if (len != 0) |
| { |
| // zero-length just means unload any previously loaded model; only look if non-zero |
| path = find_mach_model (name); |
| if (path == NULL) |
| { |
| ret = dbe_sprintf (GTXT ("Machine model %s not found\n"), name); |
| free (name); |
| return ret; |
| } |
| fptr = fopen (path, NTXT ("r")); |
| if (fptr == NULL) |
| { |
| ret = dbe_sprintf (GTXT ("Open of Machine model %s, file %s failed\n"), name, path); |
| free (path); |
| free (name); |
| return ret; |
| } |
| } |
| |
| // We are now committed to make the new machine model the loaded one; |
| // Delete any MemoryObjects from any previously loaded machinemodel |
| if (dbeSession->mach_model_loaded != NULL) |
| { |
| Vector <char *> *oldobjs = MemorySpace::getMachineModelMemObjs |
| (dbeSession->mach_model_loaded); |
| for (int i = 0; i < oldobjs->size (); i++) |
| MemorySpace::mobj_delete (oldobjs->fetch (i)); |
| delete oldobjs; |
| free (mach_model_loaded); |
| } |
| if (len == 0) |
| { |
| mach_model_loaded = NULL; |
| free (name); |
| // and there's no "loading" to do; just return |
| return NULL; |
| } |
| else |
| mach_model_loaded = name; |
| |
| int line_no = 0; |
| end_cmd = NULL; |
| |
| while (!feof (fptr)) |
| { |
| char *script = read_line (fptr); |
| if (script == NULL) |
| continue; |
| |
| line_no++; |
| strtok (script, NTXT ("\n")); |
| |
| // extract the command |
| cmd = strtok (script, NTXT (" \t")); |
| if (cmd == NULL || *cmd == '#' || *cmd == '\n') |
| { |
| free (script); |
| continue; |
| } |
| |
| char *remainder = strtok (NULL, NTXT ("\n")); |
| // now extract the arguments |
| int nargs = 0; |
| for (;;) |
| { |
| if (nargs >= MAXARGS) |
| { |
| ret = dbe_sprintf (GTXT ("Warning: more than %d arguments to %s command, line %d\n"), |
| MAXARGS, cmd, line_no); |
| continue; |
| } |
| |
| char *nextarg = strtok (remainder, NTXT ("\n")); |
| |
| if (nextarg == NULL || *nextarg == '#') |
| // either the end of the line, or a comment indicator |
| break; |
| arglist[nargs++] = parse_qstring (nextarg, &end_cmd); |
| remainder = end_cmd; |
| if (remainder == NULL) |
| break; |
| |
| // skip any blanks or tabs to get to next argument |
| while ((*remainder == ' ') || (*remainder == '\t')) |
| remainder++; |
| } |
| |
| cmd_type = Command::get_command (cmd, arg_count, cparam); |
| // check for extra arguments |
| if (cmd_type != UNKNOWN_CMD && cmd_type != INDXOBJDEF && nargs > arg_count) |
| ret = dbe_sprintf (GTXT ("Warning: extra arguments to %s command, line %d\n"), |
| cmd, line_no); |
| if (nargs < arg_count) |
| { |
| ret = dbe_sprintf (GTXT ("Error: missing arguments to %s command, line %d\n"), |
| cmd, line_no); |
| |
| // ignore this command |
| free (script); |
| continue; |
| } |
| |
| switch (cmd_type) |
| { |
| case INDXOBJDEF: |
| { |
| char *err = dbeSession->indxobj_define (arglist[0], NULL, |
| arglist[1], (nargs >= 3) ? PTXT (arglist[2]) : NULL, |
| (nargs >= 4) ? PTXT (arglist[3]) : NULL); |
| if (err != NULL) |
| ret = dbe_sprintf (GTXT (" %s: line %d `%s %s %s'\n"), |
| err, line_no, cmd, arglist[0], arglist[1]); |
| break; |
| } |
| case COMMENT: |
| // ignore the line |
| break; |
| default: |
| { |
| // unexpected command in a machinemodel file |
| ret = dbe_sprintf (GTXT ("Unexpected command in machinemodel file %s on line %d: `%.64s'\n"), |
| path, line_no, cmd); |
| break; |
| } |
| } |
| free (script); |
| } |
| fclose (fptr); |
| return ret; |
| } |
| |
| Vector<char*> * |
| DbeSession::list_mach_models () |
| { |
| Vector<char*> *model_names = new Vector<char*>(); |
| |
| // Open the current directory to read the entries there |
| DIR *dir = opendir (NTXT (".")); |
| if (dir != NULL) |
| { |
| struct dirent *entry = NULL; |
| while ((entry = readdir (dir)) != NULL) |
| { |
| size_t len = strlen (entry->d_name); |
| if (len < 5 || strcmp (entry->d_name + len - 5, ".ermm") != 0) |
| continue; |
| char *model = dbe_strdup (entry->d_name); |
| |
| // remove the .ermm suffix |
| model[len - 5] = 0; |
| |
| // add to vector |
| model_names->append (dbe_strdup (model)); |
| } |
| closedir (dir); |
| } |
| |
| // Read the user's home directory to list the models there |
| char *home = getenv ("HOME"); |
| if (home != NULL) |
| { |
| dir = opendir (home); |
| if (dir != NULL) |
| { |
| struct dirent *entry = NULL; |
| while ((entry = readdir (dir)) != NULL) |
| { |
| size_t len = strlen (entry->d_name); |
| if (len < 5 || strcmp (entry->d_name + len - 5, ".ermm") != 0) |
| continue; |
| char *model = dbe_strdup (entry->d_name); |
| |
| // remove the .ermm suffix |
| model[len - 5] = 0; |
| |
| // add to vector |
| model_names->append (dbe_strdup (model)); |
| } |
| closedir (dir); |
| } |
| } |
| |
| // Read system installation directory to list the models there |
| char *sysdir = dbe_sprintf ("%s/%s", theApplication->get_run_dir (), LIBNAME); |
| |
| dir = opendir (sysdir); |
| if (dir != NULL) |
| { |
| struct dirent *entry = NULL; |
| while ((entry = readdir (dir)) != NULL) |
| { |
| size_t len = strlen (entry->d_name); |
| if (len < 5 || strcmp (entry->d_name + len - 5, ".ermm") != 0) |
| continue; |
| char *model = dbe_strdup (entry->d_name); |
| |
| // remove the .ermm suffix |
| model[len - 5] = 0; |
| |
| // add to vector |
| model_names->append (dbe_strdup (model)); |
| } |
| closedir (dir); |
| } |
| return model_names; |
| } |