/*****************************************************************************\
 * cmos_ops.c
 * $Id: cmos_ops.c,v 1.3 2006/01/24 00:25:40 dsp_llnl Exp $
 *****************************************************************************
 *  Copyright (C) 2002-2005 The Regents of the University of California.
 *  Produced at the Lawrence Livermore National Laboratory.
 *  Written by David S. Peterson <dsp@llnl.gov> <dave_peterson@pobox.com>.
 *  UCRL-CODE-2003-012
 *  All rights reserved.
 *
 *  This file is part of lxbios, a utility for reading/writing LinuxBIOS
 *  parameters and displaying information from the LinuxBIOS table.
 *  For details, see <http://www.llnl.gov/linux/lxbios/>.
 *
 *  Please also read the file DISCLAIMER which is included in this software
 *  distribution.
 *
 *  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) version 2, dated June 1991.
 *
 *  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 terms and
 *  conditions of 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, Inc.,
 *  59 Temple Place, Suite 330, Boston, MA 02111-1307 USA.
\*****************************************************************************/

#include "common.h"
#include "cmos_ops.h"
#include "cmos_lowlevel.h"

static int prepare_cmos_op_common (const cmos_entry_t *e);

/****************************************************************************
 * prepare_cmos_op_common
 *
 * Perform a few checks common to both reads and writes.
 ****************************************************************************/
static int prepare_cmos_op_common (const cmos_entry_t *e)
 { int result;

   if (e->config == CMOS_ENTRY_RESERVED)
      /* Access to reserved parameters is not permitted. */
      return CMOS_OP_RESERVED;

   if ((result = verify_cmos_op(e->bit, e->length)) != OK)
      return result;

   assert(e->length > 0);
   return OK;
 }

/****************************************************************************
 * prepare_cmos_read
 *
 * The caller wishes to read a CMOS parameter represented by 'e'.  Perform
 * sanity checking on 'e'.  If a problem was found with e, return an error
 * code.  Else return OK.
 ****************************************************************************/
int prepare_cmos_read (const cmos_entry_t *e)
 { int result;

   if ((result = prepare_cmos_op_common(e)) != OK)
      return result;

   switch (e->config)
    { case CMOS_ENTRY_ENUM:
      case CMOS_ENTRY_HEX:
         break;

      default:
         BUG();
    }

   return OK;
 }

/****************************************************************************
 * prepare_cmos_write
 *
 * The caller wishes to set a CMOS parameter represented by 'e' to a value
 * whose string representation is stored in 'value_str'.  Perform sanity
 * checking on 'value_str'.  On error, return an error code.  Else store the
 * numeric equivalent of 'value_str' in '*value' and return OK.
 ****************************************************************************/
int prepare_cmos_write (const cmos_entry_t *e, const char value_str[],
                        unsigned long long *value)
 { const cmos_enum_t *q;
   unsigned long long out;
   const char *p;
   int negative, result, found_one;

   if ((result = prepare_cmos_op_common(e)) != OK)
      return result;

   switch (e->config)
    { case CMOS_ENTRY_ENUM:
         /* Make sure the user's input corresponds to a valid option. */
         for (q = first_cmos_enum_id(e->config_id), found_one = 0;
              q != NULL;
              q = next_cmos_enum_id(q))
          { found_one = 1;

            if (!strncmp(q->text, value_str, CMOS_MAX_TEXT_LENGTH))
               break;
          }

         if (!found_one)
            return CMOS_OP_NO_MATCHING_ENUM;

         if (q == NULL)
            return CMOS_OP_BAD_ENUM_VALUE;

         out = q->value;
         break;

      case CMOS_ENTRY_HEX:
         /* See if the first character of 'value_str' (excluding any initial
          * whitespace) is a minus sign.
          */
         for (p = value_str; isspace(*p); p++);
         negative = (*p == '-');

         out = strtoull(value_str, (char **) &p, 0);

         if (*p)
            return CMOS_OP_INVALID_INT;

         /* If we get this far, the user specified a valid integer.  However
          * we do not currently support the use of negative numbers as CMOS
          * parameter values.
          */
         if (negative)
            return CMOS_OP_NEGATIVE_INT;

         break;

      default:
         BUG();
    }

   if ((e->length < (8 * sizeof(*value))) &&
       (out >= (1ull << e->length)))
      return CMOS_OP_VALUE_TOO_WIDE;

   *value = out;
   return OK;
 }

/****************************************************************************
 * cmos_checksum_read
 *
 * Read the checksum for the LinuxBIOS parameters stored in CMOS and return
 * this value.
 ****************************************************************************/
uint16_t cmos_checksum_read (void)
 { uint16_t lo, hi;

   /* The checksum is stored in a big-endian format. */
   hi = cmos_read_byte(cmos_checksum_index);
   lo = cmos_read_byte(cmos_checksum_index + 1);
   return (hi << 8) + lo;
 }

/****************************************************************************
 * cmos_checksum_write
 *
 * Set the checksum for the LinuxBIOS parameters stored in CMOS to
 * 'checksum'.
 ****************************************************************************/
void cmos_checksum_write (uint16_t checksum)
 { unsigned char lo, hi;

   /* The checksum is stored in a big-endian format. */
   hi = (unsigned char) (checksum >> 8);
   lo = (unsigned char) (checksum & 0x00ff);
   cmos_write_byte(cmos_checksum_index, hi);
   cmos_write_byte(cmos_checksum_index + 1, lo);
 }

/****************************************************************************
 * cmos_checksum_compute
 *
 * Compute a checksum for the LinuxBIOS parameter values currently stored in
 * CMOS and return this checksum.
 ****************************************************************************/
uint16_t cmos_checksum_compute (void)
 { unsigned i, sum;

   sum = 0;

   for (i = cmos_checksum_start; i <= cmos_checksum_end; i++)
      sum += cmos_read_byte(i);

   return ~((uint16_t) (sum & 0xffff));
 }

/****************************************************************************
 * cmos_checksum_verify
 *
 * Verify that the LinuxBIOS CMOS checksum is valid.  If checksum is not
 * valid then print warning message and exit.
 ****************************************************************************/
void cmos_checksum_verify (void)
 { uint16_t computed, actual;

   set_iopl(3);
   computed = cmos_checksum_compute();
   actual = cmos_checksum_read();
   set_iopl(0);

   if (computed != actual)
    { fprintf(stderr, "%s: Warning: LinuxBIOS CMOS checksum is bad.\n",
              prog_name);
      exit(1);
    }
 }