/*  console.c -- Open Firmware console for GRUB.  */
/*
 *  GRUB  --  GRand Unified Bootloader
 *  Copyright (C) 2003,2004,2005,2007,2008,2009  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/term.h>
#include <grub/types.h>
#include <grub/misc.h>
#include <grub/mm.h>
#include <grub/time.h>
#include <grub/terminfo.h>
#include <grub/dl.h>
#include <grub/speaker.h>

GRUB_MOD_LICENSE ("GPLv3+");

extern struct grub_terminfo_output_state grub_spkmodem_terminfo_output;

static void
make_tone (grub_uint16_t freq_count, unsigned int duration)
{
  /* Program timer 2.  */
  grub_outb (GRUB_PIT_CTRL_SELECT_2
	     | GRUB_PIT_CTRL_READLOAD_WORD
	     | GRUB_PIT_CTRL_SQUAREWAVE_GEN
	     | GRUB_PIT_CTRL_COUNT_BINARY, GRUB_PIT_CTRL);
  grub_outb (freq_count & 0xff, GRUB_PIT_COUNTER_2);		/* LSB */
  grub_outb ((freq_count >> 8) & 0xff, GRUB_PIT_COUNTER_2);	/* MSB */

  /* Start speaker.  */
  grub_outb (grub_inb (GRUB_PIT_SPEAKER_PORT)
	     | GRUB_PIT_SPK_TMR2 | GRUB_PIT_SPK_DATA,
	     GRUB_PIT_SPEAKER_PORT);

  for (; duration; duration--)
    {
      unsigned short counter, previous_counter = 0xffff;
      while (1)
	{
	  counter = grub_inb (GRUB_PIT_COUNTER_2);
	  counter |= ((grub_uint16_t) grub_inb (GRUB_PIT_COUNTER_2)) << 8;
	  if (counter > previous_counter)
	    {
	      previous_counter = counter;
	      break;
	    }
	  previous_counter = counter;
	}
    }
}

static int inited;

static void
put (struct grub_term_output *term __attribute__ ((unused)), const int c)
{
  int i;

  make_tone (GRUB_SPEAKER_PIT_FREQUENCY / 200, 4);
  for (i = 7; i >= 0; i--)
    {
      if ((c >> i) & 1)
	make_tone (GRUB_SPEAKER_PIT_FREQUENCY / 2000, 20);
      else
	make_tone (GRUB_SPEAKER_PIT_FREQUENCY / 4000, 40);
      make_tone (GRUB_SPEAKER_PIT_FREQUENCY / 1000, 10);
    }
  make_tone (GRUB_SPEAKER_PIT_FREQUENCY / 200, 0);
}

static grub_err_t
grub_spkmodem_init_output (struct grub_term_output *term)
{
  /* Some models shutdown sound when not in use and it takes for it
     around 30 ms to come back on which loses 3 bits. So generate a base
     200 Hz continously. */

  make_tone (GRUB_SPEAKER_PIT_FREQUENCY / 200, 0);
  grub_terminfo_output_init (term);

  return 0;
}

static grub_err_t
grub_spkmodem_fini_output (struct grub_term_output *term __attribute__ ((unused)))
{
  grub_speaker_beep_off ();
  inited = 0;
  return 0;
}




struct grub_terminfo_output_state grub_spkmodem_terminfo_output =
  {
    .put = put,
    .width = 80,
    .height = 24
  };

static struct grub_term_output grub_spkmodem_term_output =
  {
    .name = "spkmodem",
    .init = grub_spkmodem_init_output,
    .fini = grub_spkmodem_fini_output,
    .putchar = grub_terminfo_putchar,
    .getxy = grub_terminfo_getxy,
    .getwh = grub_terminfo_getwh,
    .gotoxy = grub_terminfo_gotoxy,
    .cls = grub_terminfo_cls,
    .setcolorstate = grub_terminfo_setcolorstate,
    .setcursor = grub_terminfo_setcursor,
    .flags = GRUB_TERM_CODE_TYPE_ASCII,
    .data = &grub_spkmodem_terminfo_output,
  };

GRUB_MOD_INIT (spkmodem)
{
  grub_term_register_output ("spkmodem", &grub_spkmodem_term_output);
}


GRUB_MOD_FINI (spkmodem)
{
  grub_term_unregister_output (&grub_spkmodem_term_output);
}