sleep.c 2.65 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21
/* sleep.c - Command to wait a specified number of seconds.  */
/*
 *  GRUB  --  GRand Unified Bootloader
 *  Copyright (C) 2006,2007,2008  Free Software Foundation, Inc.
 *
 *  GRUB 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 of the License, or
 *  (at your option) any later version.
 *
 *  GRUB 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 GRUB.  If not, see <http://www.gnu.org/licenses/>.
 */

#include <grub/dl.h>
#include <grub/term.h>
22
#include <grub/time.h>
23 24
#include <grub/types.h>
#include <grub/misc.h>
25
#include <grub/extcmd.h>
26
#include <grub/i18n.h>
27

28 29
GRUB_MOD_LICENSE ("GPLv3+");

30 31
static const struct grub_arg_option options[] =
  {
32
    {"verbose", 'v', 0, N_("Verbose countdown."), 0, 0},
33
    {"interruptible", 'i', 0, N_("Allow to interrupt with ESC."), 0, 0},
34 35 36
    {0, 0, 0, 0, 0, 0}
  };

37
static struct grub_term_coordinate *pos;
38 39 40 41

static void
do_print (int n)
{
42
  grub_term_restore_pos (pos);
43 44 45
  /* NOTE: Do not remove the trailing space characters.
     They are required to clear the line.  */
  grub_printf ("%d    ", n);
46
  grub_refresh ();
47 48
}

49
/* Based on grub_millisleep() from kern/generic/millisleep.c.  */
50 51 52
static int
grub_interruptible_millisleep (grub_uint32_t ms)
{
53 54 55 56 57
  grub_uint64_t start;

  start = grub_get_time_ms ();

  while (grub_get_time_ms () - start < ms)
58
    if (grub_getkey_noblock () == GRUB_TERM_ESC)
59 60 61 62 63 64
      return 1;

  return 0;
}

static grub_err_t
65
grub_cmd_sleep (grub_extcmd_context_t ctxt, int argc, char **args)
66
{
67
  struct grub_arg_list *state = ctxt->state;
68 69 70
  int n;

  if (argc != 1)
71
    return grub_error (GRUB_ERR_BAD_ARGUMENT, N_("one argument expected"));
72 73 74 75 76 77 78 79 80

  n = grub_strtoul (args[0], 0, 10);

  if (n == 0)
    {
      /* Either `0' or broken input.  */
      return 0;
    }

81 82
  grub_refresh ();

83
  pos = grub_term_save_pos ();
84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103

  for (; n; n--)
    {
      if (state[0].set)
	do_print (n);

      if (state[1].set)
	{
	  if (grub_interruptible_millisleep (1000))
	    return 1;
	}
      else
	grub_millisleep (1000);
    }
  if (state[0].set)
    do_print (0);

  return 0;
}

104
static grub_extcmd_t cmd;
105 106 107

GRUB_MOD_INIT(sleep)
{
108
  cmd = grub_register_extcmd ("sleep", grub_cmd_sleep, 0,
109 110
			      N_("NUMBER_OF_SECONDS"),
			      N_("Wait for a specified number of seconds."),
111
			      options);
112 113 114 115
}

GRUB_MOD_FINI(sleep)
{
116
  grub_unregister_extcmd (cmd);
117
}