ACC SHELL

Path : /proc/self/root/usr/share/slsh/scripts/
File Upload :
Current File : //proc/self/root/usr/share/slsh/scripts/slprof

#!/usr/bin/env slsh
require ("profile");
require ("cmdopt");

private variable Version = "0.2.2-0";
private variable Profile_Fp;

private define _slprof_version ()
{
   () = fprintf (stderr, "slprof version %s; S-Lang version: %s\n",
		 Version, _slang_version_string);
}

private define _slprof_usage ()
{
   variable fp = stderr;
   () = fprintf (fp, "Usage: %s [options] script args...\n", path_basename (__argv[0]));
   () = fprintf (fp, "Options:\n");
   variable opts = 
     [
      " -c|--calibrate <N>         Calibrate using N repetitions [default:1000, 0=off]\n",
      " -f|--func <function>       Name of Function to call [default: slsh_main]\n",
      " -l|--lines                 Profile lines, not just functions\n",
      " -o|--output <file>         Name of output file [default: <script>.slprof]\n",
      " -v|--version               Print version information\n",
      " -h|--help                  Print this message\n"
     ];
   variable opt;
   foreach opt (opts)
     () = fputs (opt, fp);
   exit (1);
}

private define _slprof_cmdopt_error (msg)
{
   () = fprintf (stderr, "%s\n", msg);
   _slprof_usage ();
}

private define _slprof_write_profile ()
{
   profile_report (Profile_Fp);
   () = fclose (Profile_Fp);
}

private define _slprof_calibrate (n)
{
   if (n == 0)
     return;
   () = fprintf (stdout, "Calibrating profiler..."); ()=fflush (stdout);
   profile_calibrate (n);
   () = fprintf (stdout, "Done\n");
}

private define _slprof_main ()
{
   variable output = NULL;
   variable func = "slsh_main";
   variable line_by_line = 0;
   variable cal_n = 1000;

   variable opts = cmdopt_new (&_slprof_cmdopt_error);
   opts.add("c|calibrate", &cal_n; type="int");
   opts.add("f|funct", &func; type="string");
   opts.add("l|lines", &line_by_line);
   opts.add("o|output",&output; type="string");
   opts.add("v|version", &_slprof_version),
   opts.add("h|help", &_slprof_usage);
   variable i = opts.process (__argv, 1);

   if (cal_n < 0) cal_n = 0;
   if (func == "")
     _slprof_usage ();
   
   if (i >= __argc)
     _slprof_usage ();

   variable main = strtok (func, " \t(;")[0];
   variable main_args = strtrim (substrbytes (func, 1+strlen(main), -1));
   
   variable depth = _stkdepth();
   try
     {
	if (strlen (main)) eval (main_args);
     }
   catch AnyError:
     {
	() = fprintf (stderr, "Error parsing args of %s\n", func);
	exit (1);
     }
   main_args = __pop_args (_stkdepth ()-depth);

   variable script = __argv[i];

   if (not path_is_absolute (script))
     script = path_concat (getcwd (), script);
   
   variable st = stat_file (script);
   if (st == NULL)
     {
	() = fprintf (stderr, "Unable to stat %s -- did you specify its path?\n", script);
	exit (1);
     }

   if (not stat_is ("reg", st.st_mode))
     () = fprintf (stderr, "*** Warning %s is not a regular file\n", script);

   if (output == NULL)
     output = strcat (path_basename_sans_extname (script), ".slprof");

   variable fp = fopen (output, "w");
   if (fp == NULL)
     {
	() = fprintf (stderr, "Unable to profiler output file %s\n");
	exit (1);
     }
   () = fprintf (fp, "# slprof profile report for:\n");
   () = fprintf (fp, "#  %S\n\n", strjoin (__argv, " "));
   Profile_Fp = fp;

   variable has_at_exit = 0;
#ifexists atexit
   atexit (&_slprof_write_profile);
   has_at_exit = 1;
#endif

   __set_argc_argv (__argv[[i:]]);
	
   % Actual profiling starts here

   _slprof_calibrate (cal_n);
   profile_begin (line_by_line);
   () = evalfile (script);

   variable ref = __get_reference (main);
   if (ref != NULL)
     (@ref) (__push_args (main_args));
   else if (main != "slsh_main")
     () = fprintf (stderr, "*** Warning: %s is not defined\n", main);

   profile_end ();

   if (has_at_exit == 0)
     _slprof_write_profile ();
}

_slprof_main ();
exit (0);

ACC SHELL 2018