2 * yosys -- Yosys Open SYnthesis Suite
4 * Copyright (C) 2012 Clifford Wolf <clifford@clifford.at>
6 * Permission to use, copy, modify, and/or distribute this software for any
7 * purpose with or without fee is hereby granted, provided that the above
8 * copyright notice and this permission notice appear in all copies.
10 * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
11 * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
12 * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
13 * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
14 * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
15 * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
16 * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
20 #include "kernel/yosys.h"
21 #include "kernel/celltypes.h"
23 #ifdef YOSYS_ENABLE_READLINE
24 # include <readline/readline.h>
25 # include <readline/history.h>
28 #ifdef YOSYS_ENABLE_PLUGINS
35 #elif defined(__APPLE__)
36 # include <mach-o/dyld.h>
39 # include <sys/stat.h>
43 # include <sys/types.h>
44 # include <sys/stat.h>
54 RTLIL::Design
*yosys_design
= NULL
;
55 CellTypes yosys_celltypes
;
57 #ifdef YOSYS_ENABLE_TCL
58 Tcl_Interp
*yosys_tcl_interp
= NULL
;
61 bool memhasher_active
= false;
62 uint32_t memhasher_rng
= 123456;
63 std::vector
<void*> memhasher_store
;
68 memhasher_rng
+= time(NULL
) << 16 ^ getpid();
70 memhasher_store
.resize(0x10000);
71 memhasher_active
= true;
76 for (auto p
: memhasher_store
)
78 memhasher_store
.clear();
79 memhasher_active
= false;
84 memhasher_rng
^= memhasher_rng
<< 13;
85 memhasher_rng
^= memhasher_rng
>> 17;
86 memhasher_rng
^= memhasher_rng
<< 5;
88 int size
, index
= (memhasher_rng
>> 4) & 0xffff;
89 switch (memhasher_rng
& 7) {
90 case 0: size
= 16; break;
91 case 1: size
= 256; break;
92 case 2: size
= 1024; break;
93 case 3: size
= 4096; break;
96 if (index
< 16) size
*= 16;
97 memhasher_store
[index
] = realloc(memhasher_store
[index
], size
);
100 std::string
stringf(const char *fmt
, ...)
106 string
= vstringf(fmt
, ap
);
112 std::string
vstringf(const char *fmt
, va_list ap
)
122 str
= (char*)realloc(str
, sz
);
123 rc
= vsnprintf(str
, sz
, fmt
, apc
);
125 if (rc
>= 0 && rc
< sz
)
130 if (vasprintf(&str
, fmt
, ap
) < 0)
142 int readsome(std::istream
&f
, char *s
, int n
)
144 int rc
= f
.readsome(s
, n
);
146 // f.readsome() sometimes returns 0 on a non-empty stream..
158 std::string
next_token(std::string
&text
, const char *sep
)
160 size_t pos_begin
= text
.find_first_not_of(sep
);
162 if (pos_begin
== std::string::npos
)
163 pos_begin
= text
.size();
165 size_t pos_end
= text
.find_first_of(sep
, pos_begin
);
167 if (pos_end
== std::string::npos
)
168 pos_end
= text
.size();
170 std::string token
= text
.substr(pos_begin
, pos_end
-pos_begin
);
171 text
= text
.substr(pos_end
);
175 // this is very similar to fnmatch(). the exact rules used by this
178 // ? matches any character except
179 // * matches any sequence of characters
180 // [...] matches any of the characters in the list
181 // [!..] matches any of the characters not in the list
183 // a backslash may be used to escape the next characters in the
184 // pattern. each special character can also simply match itself.
186 bool patmatch(const char *pattern
, const char *string
)
191 if (*pattern
== '\\') {
192 if (pattern
[1] == string
[0] && patmatch(pattern
+2, string
+1))
196 if (*pattern
== '?') {
199 return patmatch(pattern
+1, string
+1);
202 if (*pattern
== '*') {
204 if (patmatch(pattern
+1, string
++))
207 return pattern
[1] == 0;
210 if (*pattern
== '[') {
211 bool found_match
= false;
212 bool inverted_list
= pattern
[1] == '!';
213 const char *p
= pattern
+ (inverted_list
? 1 : 0);
217 if (found_match
!= inverted_list
&& patmatch(p
+1, string
+1))
231 if (*pattern
== *string
)
232 return patmatch(pattern
+1, string
+1);
237 int run_command(const std::string
&command
, std::function
<void(const std::string
&)> process_line
)
240 return system(command
.c_str());
242 FILE *f
= popen(command
.c_str(), "r");
248 while (fgets(logbuf
, 128, f
) != NULL
) {
250 if (!line
.empty() && line
.back() == '\n')
251 process_line(line
), line
.clear();
262 return WEXITSTATUS(ret
);
266 std::string
make_temp_file(std::string template_str
)
269 if (template_str
.rfind("/tmp/", 0) == 0) {
271 char longpath
[MAX_PATH
+ 1];
272 char shortpath
[MAX_PATH
+ 1];
274 WCHAR longpath
[MAX_PATH
+ 1];
275 TCHAR shortpath
[MAX_PATH
+ 1];
277 if (!GetTempPath(MAX_PATH
+1, longpath
))
278 log_error("GetTempPath() failed.\n");
279 if (!GetShortPathName(longpath
, shortpath
, MAX_PATH
+ 1))
280 log_error("GetShortPathName() failed.\n");
282 for (int i
= 0; shortpath
[i
]; i
++)
283 path
+= char(shortpath
[i
]);
284 template_str
= stringf("%s\\%s", path
.c_str(), template_str
.c_str() + 5);
287 size_t pos
= template_str
.rfind("XXXXXX");
288 log_assert(pos
!= std::string::npos
);
291 for (int i
= 0; i
< 6; i
++) {
292 static std::string y
= "0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ";
293 static uint32_t x
= 314159265 ^ uint32_t(time(NULL
));
294 x
^= x
<< 13, x
^= x
>> 17, x
^= x
<< 5;
295 template_str
[pos
+i
] = y
[x
% y
.size()];
297 if (_access(template_str
.c_str(), 0) != 0)
301 size_t pos
= template_str
.rfind("XXXXXX");
302 log_assert(pos
!= std::string::npos
);
304 int suffixlen
= GetSize(template_str
) - pos
- 6;
306 char *p
= strdup(template_str
.c_str());
307 close(mkstemps(p
, suffixlen
));
315 std::string
make_temp_dir(std::string template_str
)
318 template_str
= make_temp_file(template_str
);
319 mkdir(template_str
.c_str());
322 size_t pos
= template_str
.rfind("XXXXXX");
323 log_assert(pos
!= std::string::npos
);
325 int suffixlen
= GetSize(template_str
) - pos
- 6;
326 log_assert(suffixlen
== 0);
328 char *p
= strdup(template_str
.c_str());
330 log_assert(p
!= NULL
);
339 bool check_file_exists(std::string filename
, bool)
341 return _access(filename
.c_str(), 0) == 0;
344 bool check_file_exists(std::string filename
, bool is_exec
)
346 return access(filename
.c_str(), is_exec
? X_OK
: F_OK
) == 0;
350 void remove_directory(std::string dirname
)
353 run_command(stringf("rmdir /s /q \"%s\"", dirname
.c_str()));
356 struct dirent
**namelist
;
357 int n
= scandir(dirname
.c_str(), &namelist
, nullptr, alphasort
);
359 for (int i
= 0; i
< n
; i
++) {
360 if (strcmp(namelist
[i
]->d_name
, ".") && strcmp(namelist
[i
]->d_name
, "..")) {
361 std::string buffer
= stringf("%s/%s", dirname
.c_str(), namelist
[i
]->d_name
);
362 if (!stat(buffer
.c_str(), &stbuf
) && S_ISREG(stbuf
.st_mode
)) {
363 remove(buffer
.c_str());
365 remove_directory(buffer
);
370 rmdir(dirname
.c_str());
374 int GetSize(RTLIL::Wire
*wire
)
381 // if there are already IdString objects then we have a global initialization order bug
383 log_assert(empty_id
.index_
== 0);
384 IdString::get_reference(empty_id
.index_
);
386 Pass::init_register();
387 yosys_design
= new RTLIL::Design
;
388 yosys_celltypes
.setup();
392 void yosys_shutdown()
399 for (auto f
: log_files
)
405 Pass::done_register();
406 yosys_celltypes
.clear();
408 #ifdef YOSYS_ENABLE_TCL
409 if (yosys_tcl_interp
!= NULL
) {
410 Tcl_DeleteInterp(yosys_tcl_interp
);
412 yosys_tcl_interp
= NULL
;
416 #ifdef YOSYS_ENABLE_PLUGINS
417 for (auto &it
: loaded_plugins
)
420 loaded_plugins
.clear();
421 loaded_plugin_aliases
.clear();
425 IdString::put_reference(empty_id
.index_
);
428 RTLIL::IdString
new_id(std::string file
, int line
, std::string func
)
431 size_t pos
= file
.find_last_of("/\\");
433 size_t pos
= file
.find_last_of('/');
435 if (pos
!= std::string::npos
)
436 file
= file
.substr(pos
+1);
438 pos
= func
.find_last_of(':');
439 if (pos
!= std::string::npos
)
440 func
= func
.substr(pos
+1);
442 return stringf("$auto$%s:%d:%s$%d", file
.c_str(), line
, func
.c_str(), autoidx
++);
445 RTLIL::Design
*yosys_get_design()
450 const char *create_prompt(RTLIL::Design
*design
, int recursion_counter
)
452 static char buffer
[100];
453 std::string str
= "\n";
454 if (recursion_counter
> 1)
455 str
+= stringf("(%d) ", recursion_counter
);
457 if (!design
->selected_active_module
.empty())
458 str
+= stringf(" [%s]", RTLIL::unescape_id(design
->selected_active_module
).c_str());
459 if (!design
->selection_stack
.empty() && !design
->selection_stack
.back().full_selection
) {
460 if (design
->selected_active_module
.empty())
462 else if (design
->selection_stack
.back().selected_modules
.size() != 1 || design
->selection_stack
.back().selected_members
.size() != 0 ||
463 design
->selection_stack
.back().selected_modules
.count(design
->selected_active_module
) == 0)
466 snprintf(buffer
, 100, "%s> ", str
.c_str());
470 #ifdef YOSYS_ENABLE_TCL
471 static int tcl_yosys_cmd(ClientData
, Tcl_Interp
*interp
, int argc
, const char *argv
[])
473 std::vector
<std::string
> args
;
474 for (int i
= 1; i
< argc
; i
++)
475 args
.push_back(argv
[i
]);
477 if (args
.size() >= 1 && args
[0] == "-import") {
478 for (auto &it
: pass_register
) {
479 std::string tcl_command_name
= it
.first
;
480 if (tcl_command_name
== "proc")
481 tcl_command_name
= "procs";
483 if (Tcl_GetCommandInfo(interp
, tcl_command_name
.c_str(), &info
) != 0) {
484 log("[TCL: yosys -import] Command name collision: found pre-existing command `%s' -> skip.\n", it
.first
.c_str());
486 std::string tcl_script
= stringf("proc %s args { yosys %s {*}$args }", tcl_command_name
.c_str(), it
.first
.c_str());
487 Tcl_Eval(interp
, tcl_script
.c_str());
493 if (args
.size() == 1) {
494 Pass::call(yosys_get_design(), args
[0]);
498 Pass::call(yosys_get_design(), args
);
502 extern Tcl_Interp
*yosys_get_tcl_interp()
504 if (yosys_tcl_interp
== NULL
) {
505 yosys_tcl_interp
= Tcl_CreateInterp();
506 Tcl_CreateCommand(yosys_tcl_interp
, "yosys", tcl_yosys_cmd
, NULL
, NULL
);
508 return yosys_tcl_interp
;
511 struct TclPass
: public Pass
{
512 TclPass() : Pass("tcl", "execute a TCL script file") { }
513 virtual void help() {
515 log(" tcl <filename>\n");
517 log("This command executes the tcl commands in the specified file.\n");
518 log("Use 'yosys cmd' to run the yosys command 'cmd' from tcl.\n");
520 log("The tcl command 'yosys -import' can be used to import all yosys\n");
521 log("commands directly as tcl commands to the tcl shell. The yosys\n");
522 log("command 'proc' is wrapped using the tcl command 'procs' in order\n");
523 log("to avoid a name collision with the tcl builting command 'proc'.\n");
526 virtual void execute(std::vector
<std::string
> args
, RTLIL::Design
*design
) {
528 log_cmd_error("Missing script file.\n");
530 extra_args(args
, 1, design
, false);
531 if (Tcl_EvalFile(yosys_get_tcl_interp(), args
[1].c_str()) != TCL_OK
)
532 log_cmd_error("TCL interpreter returned an error: %s\n", Tcl_GetStringResult(yosys_get_tcl_interp()));
537 #if defined(__linux__)
538 std::string
proc_self_dirname()
541 ssize_t buflen
= readlink("/proc/self/exe", path
, sizeof(path
));
543 log_error("readlink(\"/proc/self/exe\") failed: %s\n", strerror(errno
));
545 while (buflen
> 0 && path
[buflen
-1] != '/')
547 return std::string(path
, buflen
);
549 #elif defined(__APPLE__)
550 std::string
proc_self_dirname()
554 while (_NSGetExecutablePath(path
, &buflen
) != 0)
555 path
= (char *) realloc((void *) path
, buflen
);
556 while (buflen
> 0 && path
[buflen
-1] != '/')
558 return std::string(path
, buflen
);
560 #elif defined(_WIN32)
561 std::string
proc_self_dirname()
565 char longpath
[MAX_PATH
+ 1];
566 char shortpath
[MAX_PATH
+ 1];
568 WCHAR longpath
[MAX_PATH
+ 1];
569 TCHAR shortpath
[MAX_PATH
+ 1];
571 if (!GetModuleFileName(0, longpath
, MAX_PATH
+1))
572 log_error("GetModuleFileName() failed.\n");
573 if (!GetShortPathName(longpath
, shortpath
, MAX_PATH
+1))
574 log_error("GetShortPathName() failed.\n");
575 while (shortpath
[i
] != 0)
577 while (i
> 0 && shortpath
[i
-1] != '/' && shortpath
[i
-1] != '\\')
580 for (i
= 0; shortpath
[i
]; i
++)
581 path
+= char(shortpath
[i
]);
584 #elif defined(EMSCRIPTEN)
585 std::string
proc_self_dirname()
590 #error Dont know how to determine process executable base path!
593 std::string
proc_share_dirname()
595 std::string proc_self_path
= proc_self_dirname();
597 std::string proc_share_path
= proc_self_path
+ "share\\";
598 if (check_file_exists(proc_share_path
, true))
599 return proc_share_path
;
600 proc_share_path
= proc_self_path
+ "..\\share\\";
601 if (check_file_exists(proc_share_path
, true))
602 return proc_share_path
;
604 std::string proc_share_path
= proc_self_path
+ "share/";
605 if (check_file_exists(proc_share_path
, true))
606 return proc_share_path
;
607 proc_share_path
= proc_self_path
+ "../share/yosys/";
608 if (check_file_exists(proc_share_path
, true))
609 return proc_share_path
;
611 log_error("proc_share_dirname: unable to determine share/ directory!\n");
614 bool fgetline(FILE *f
, std::string
&buffer
)
619 if (fgets(block
, 4096, f
) == NULL
)
622 if (buffer
.size() > 0 && (buffer
[buffer
.size()-1] == '\n' || buffer
[buffer
.size()-1] == '\r')) {
623 while (buffer
.size() > 0 && (buffer
[buffer
.size()-1] == '\n' || buffer
[buffer
.size()-1] == '\r'))
624 buffer
.resize(buffer
.size()-1);
630 static void handle_label(std::string
&command
, bool &from_to_active
, const std::string
&run_from
, const std::string
&run_to
)
635 while (pos
< GetSize(command
) && (command
[pos
] == ' ' || command
[pos
] == '\t'))
638 while (pos
< GetSize(command
) && command
[pos
] != ' ' && command
[pos
] != '\t' && command
[pos
] != '\r' && command
[pos
] != '\n')
639 label
+= command
[pos
++];
641 if (label
.back() == ':' && GetSize(label
) > 1)
643 label
= label
.substr(0, GetSize(label
)-1);
644 command
= command
.substr(pos
);
646 if (label
== run_from
)
647 from_to_active
= true;
648 else if (label
== run_to
|| (run_from
== run_to
&& !run_from
.empty()))
649 from_to_active
= false;
653 void run_frontend(std::string filename
, std::string command
, RTLIL::Design
*design
, std::string
*backend_command
, std::string
*from_to_label
)
655 if (command
== "auto") {
656 if (filename
.size() > 2 && filename
.substr(filename
.size()-2) == ".v")
658 else if (filename
.size() > 2 && filename
.substr(filename
.size()-3) == ".sv")
659 command
= "verilog -sv";
660 else if (filename
.size() > 3 && filename
.substr(filename
.size()-3) == ".il")
662 else if (filename
.size() > 3 && filename
.substr(filename
.size()-3) == ".ys")
664 else if (filename
== "-")
667 log_error("Can't guess frontend for input file `%s' (missing -f option)!\n", filename
.c_str());
670 if (command
== "script")
672 std::string run_from
, run_to
;
673 bool from_to_active
= true;
675 if (from_to_label
!= NULL
) {
676 size_t pos
= from_to_label
->find(':');
677 if (pos
== std::string::npos
) {
678 run_from
= *from_to_label
;
679 run_to
= *from_to_label
;
681 run_from
= from_to_label
->substr(0, pos
);
682 run_to
= from_to_label
->substr(pos
+1);
684 from_to_active
= run_from
.empty();
687 log("\n-- Executing script file `%s' --\n", filename
.c_str());
692 f
= fopen(filename
.c_str(), "r");
695 log_error("Can't open script file `%s' for reading: %s\n", filename
.c_str(), strerror(errno
));
697 FILE *backup_script_file
= Frontend::current_script_file
;
698 Frontend::current_script_file
= f
;
702 while (fgetline(f
, command
)) {
703 while (!command
.empty() && command
[command
.size()-1] == '\\') {
704 std::string next_line
;
705 if (!fgetline(f
, next_line
))
707 command
.resize(command
.size()-1);
708 command
+= next_line
;
710 handle_label(command
, from_to_active
, run_from
, run_to
);
712 Pass::call(design
, command
);
715 if (!command
.empty()) {
716 handle_label(command
, from_to_active
, run_from
, run_to
);
718 Pass::call(design
, command
);
721 catch (log_cmd_error_exception
) {
722 Frontend::current_script_file
= backup_script_file
;
723 throw log_cmd_error_exception();
726 Frontend::current_script_file
= backup_script_file
;
731 if (backend_command
!= NULL
&& *backend_command
== "auto")
732 *backend_command
= "";
737 if (filename
== "-") {
738 log("\n-- Parsing stdin using frontend `%s' --\n", command
.c_str());
740 log("\n-- Parsing `%s' using frontend `%s' --\n", filename
.c_str(), command
.c_str());
743 Frontend::frontend_call(design
, NULL
, filename
, command
);
746 void run_pass(std::string command
, RTLIL::Design
*design
)
748 log("\n-- Running pass `%s' --\n", command
.c_str());
750 Pass::call(design
, command
);
753 void run_backend(std::string filename
, std::string command
, RTLIL::Design
*design
)
755 if (command
== "auto") {
756 if (filename
.size() > 2 && filename
.substr(filename
.size()-2) == ".v")
758 else if (filename
.size() > 3 && filename
.substr(filename
.size()-3) == ".il")
760 else if (filename
.size() > 5 && filename
.substr(filename
.size()-5) == ".blif")
762 else if (filename
== "-")
764 else if (filename
.empty())
767 log_error("Can't guess backend for output file `%s' (missing -b option)!\n", filename
.c_str());
770 if (filename
.empty())
773 if (filename
== "-") {
774 log("\n-- Writing to stdout using backend `%s' --\n", command
.c_str());
776 log("\n-- Writing to `%s' using backend `%s' --\n", filename
.c_str(), command
.c_str());
779 Backend::backend_call(design
, NULL
, filename
, command
);
782 #ifdef YOSYS_ENABLE_READLINE
783 static char *readline_cmd_generator(const char *text
, int state
)
785 static std::map
<std::string
, Pass
*>::iterator it
;
789 it
= pass_register
.begin();
793 for (; it
!= pass_register
.end(); it
++) {
794 if (it
->first
.substr(0, len
) == text
)
795 return strdup((it
++)->first
.c_str());
800 static char *readline_obj_generator(const char *text
, int state
)
802 static std::vector
<char*> obj_names
;
810 RTLIL::Design
*design
= yosys_get_design();
811 int len
= strlen(text
);
813 if (design
->selected_active_module
.empty())
815 for (auto &it
: design
->modules_
)
816 if (RTLIL::unescape_id(it
.first
).substr(0, len
) == text
)
817 obj_names
.push_back(strdup(RTLIL::id2cstr(it
.first
)));
820 if (design
->modules_
.count(design
->selected_active_module
) > 0)
822 RTLIL::Module
*module
= design
->modules_
.at(design
->selected_active_module
);
824 for (auto &it
: module
->wires_
)
825 if (RTLIL::unescape_id(it
.first
).substr(0, len
) == text
)
826 obj_names
.push_back(strdup(RTLIL::id2cstr(it
.first
)));
828 for (auto &it
: module
->memories
)
829 if (RTLIL::unescape_id(it
.first
).substr(0, len
) == text
)
830 obj_names
.push_back(strdup(RTLIL::id2cstr(it
.first
)));
832 for (auto &it
: module
->cells_
)
833 if (RTLIL::unescape_id(it
.first
).substr(0, len
) == text
)
834 obj_names
.push_back(strdup(RTLIL::id2cstr(it
.first
)));
836 for (auto &it
: module
->processes
)
837 if (RTLIL::unescape_id(it
.first
).substr(0, len
) == text
)
838 obj_names
.push_back(strdup(RTLIL::id2cstr(it
.first
)));
841 std::sort(obj_names
.begin(), obj_names
.end());
844 if (idx
< obj_names
.size())
845 return strdup(obj_names
[idx
++]);
852 static char **readline_completion(const char *text
, int start
, int)
855 return rl_completion_matches(text
, readline_cmd_generator
);
856 if (strncmp(rl_line_buffer
, "read_", 5) && strncmp(rl_line_buffer
, "write_", 6))
857 return rl_completion_matches(text
, readline_obj_generator
);
862 void shell(RTLIL::Design
*design
)
864 static int recursion_counter
= 0;
867 log_cmd_error_throw
= true;
869 #ifdef YOSYS_ENABLE_READLINE
870 rl_readline_name
= "yosys";
871 rl_attempted_completion_function
= readline_completion
;
872 rl_basic_word_break_characters
= " \t\n";
875 char *command
= NULL
;
876 #ifdef YOSYS_ENABLE_READLINE
877 while ((command
= readline(create_prompt(design
, recursion_counter
))) != NULL
)
880 char command_buffer
[4096];
883 fputs(create_prompt(design
, recursion_counter
), stdout
);
885 if ((command
= fgets(command_buffer
, 4096, stdin
)) == NULL
)
888 if (command
[strspn(command
, " \t\r\n")] == 0)
890 #ifdef YOSYS_ENABLE_READLINE
891 add_history(command
);
894 char *p
= command
+ strspn(command
, " \t\r\n");
895 if (!strncmp(p
, "exit", 4)) {
897 p
+= strspn(p
, " \t\r\n");
903 log_assert(design
->selection_stack
.size() == 1);
904 Pass::call(design
, command
);
905 } catch (log_cmd_error_exception
) {
906 while (design
->selection_stack
.size() > 1)
907 design
->selection_stack
.pop_back();
915 log_cmd_error_throw
= false;
918 struct ShellPass
: public Pass
{
919 ShellPass() : Pass("shell", "enter interactive command mode") { }
920 virtual void help() {
924 log("This command enters the interactive command mode. This can be useful\n");
925 log("in a script to interrupt the script at a certain point and allow for\n");
926 log("interactive inspection or manual synthesis of the design at this point.\n");
928 log("The command prompt of the interactive shell indicates the current\n");
929 log("selection (see 'help select'):\n");
932 log(" the entire design is selected\n");
935 log(" only part of the design is selected\n");
937 log(" yosys [modname]>\n");
938 log(" the entire module 'modname' is selected using 'select -module modname'\n");
940 log(" yosys [modname]*>\n");
941 log(" only part of current module 'modname' is selected\n");
943 log("When in interactive shell, some errors (e.g. invalid command arguments)\n");
944 log("do not terminate yosys but return to the command prompt.\n");
946 log("This command is the default action if nothing else has been specified\n");
947 log("on the command line.\n");
949 log("Press Ctrl-D or type 'exit' to leave the interactive shell.\n");
952 virtual void execute(std::vector
<std::string
> args
, RTLIL::Design
*design
) {
953 extra_args(args
, 1, design
, false);
958 #ifdef YOSYS_ENABLE_READLINE
959 struct HistoryPass
: public Pass
{
960 HistoryPass() : Pass("history", "show last interactive commands") { }
961 virtual void help() {
965 log("This command prints all commands in the shell history buffer. This are\n");
966 log("all commands executed in an interactive session, but not the commands\n");
967 log("from executed scripts.\n");
970 virtual void execute(std::vector
<std::string
> args
, RTLIL::Design
*design
) {
971 extra_args(args
, 1, design
, false);
972 for(HIST_ENTRY
**list
= history_list(); *list
!= NULL
; list
++)
973 log("%s\n", (*list
)->line
);
978 struct ScriptPass
: public Pass
{
979 ScriptPass() : Pass("script", "execute commands from script file") { }
980 virtual void help() {
982 log(" script <filename> [<from_label>:<to_label>]\n");
984 log("This command executes the yosys commands in the specified file.\n");
986 log("The 2nd argument can be used to only execute the section of the\n");
987 log("file between the specified labels. An empty from label is synonymous\n");
988 log("for the beginning of the file and an empty to label is synonymous\n");
989 log("for the end of the file.\n");
991 log("If only one label is specified (without ':') then only the block\n");
992 log("marked with that label (until the next label) is executed.\n");
995 virtual void execute(std::vector
<std::string
> args
, RTLIL::Design
*design
) {
997 log_cmd_error("Missing script file.\n");
998 else if (args
.size() == 2)
999 run_frontend(args
[1], "script", design
, NULL
, NULL
);
1000 else if (args
.size() == 3)
1001 run_frontend(args
[1], "script", design
, NULL
, &args
[2]);
1003 extra_args(args
, 2, design
, false);