Skip to content
eeprom.c 10.4 KiB
Newer Older
f4grx's avatar
f4grx committed
/****************************************************************************
 * hn70ap/apps/libhn70ap/eeprom.c
 *
 *   Copyright (C) 2018 Sebastien Lorquet. All rights reserved.
 *   Author: Sebastien Lorquet <sebastien@lorquet.fr>
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions
 * are met:
 *
 * 1. Redistributions of source code must retain the above copyright
 *    notice, this list of conditions and the following disclaimer.
 * 2. Redistributions in binary form must reproduce the above copyright
 *    notice, this list of conditions and the following disclaimer in
 *    the documentation and/or other materials provided with the
 *    distribution.
 * 3. Neither the name NuttX nor the names of its contributors may be
 *    used to endorse or promote products derived from this software
 *    without specific prior written permission.
 *
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
 * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
 * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
 * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS
 * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
 * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
 * ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
 * POSSIBILITY OF SUCH DAMAGE.
 *
 ****************************************************************************/

/* EEPROM configuration management */

f4grx's avatar
f4grx committed
/****************************************************************************
 * Included Files
 ****************************************************************************/

#include <nuttx/config.h>

#include <stdio.h>
#include <string.h>

f4grx's avatar
f4grx committed
#include <fcntl.h>
#include <unistd.h>

f4grx's avatar
f4grx committed
#include <netinet/in.h>

#include <hn70ap/eeprom.h>

struct param_s
{
  const char *name; /* Config name */
  int type;         /* Config type */
  int addr;         /* Storage address of first byte in EEPROM */
f4grx's avatar
f4grx committed
  int mask;         /* For boolean configs, mask value of bit config */
f4grx's avatar
f4grx committed
};

/* EEPROM configuration
 * Address length Default Name Type Desciption
 * 0     : Config validation marker. If FF, EEPROM is virgin, and default config is loaded
 * 1-127 : Config entries described below
 */

const struct param_s eeconfig_params[] = 
{
f4grx's avatar
f4grx committed
  {"dhcp", EECONFIG_TYPE_BOOL, 0x01, 0x01}, /* 0x01 - 0x01 DHCP Client Enable */
                                            /* 0x02 - 0x03 RFU for more network options */
  {"ip"  , EECONFIG_TYPE_IP  , 0x04,   0},  /* 0x04 - 0x07 Static IP address to use if DHCP is not enabled */
  {"mask", EECONFIG_TYPE_IP  , 0x08,   0},  /* 0x08 - 0x0B Static IP mask to use if DHCP is not enabled */
  {"gw"  , EECONFIG_TYPE_IP  , 0x0C,   0},  /* 0x0C - 0x0F IP address of default gateway */
  {"dns" , EECONFIG_TYPE_IP  , 0x10,   0},  /* 0x10 - 0x13 IP address of DNS server (independent of DHCP enable)*/
  {"call", EECONFIG_TYPE_CALL, 0x14,   0},  /* 0x14 - 0x1B HAM callsign, zero padded */
  {"ssid", EECONFIG_TYPE_BYTE, 0x1C,   0},  /* 0x1C - 0x1C HAM Station ID */
                                            /* 0x1D - 0x7F RFU for more options */
f4grx's avatar
f4grx committed
};

#define EECONFIG_PARAMS_COUNT (sizeof(eeconfig_params) / sizeof(eeconfig_params[0]))

f4grx's avatar
f4grx committed
#define HN70AP_CONFIG_INSTALLED 0x42 //Vlue set at address zero of eeprom if
f4grx's avatar
f4grx committed
/*----------------------------------------------------------------------------*/
f4grx's avatar
f4grx committed
int hn70ap_eeconfig_init(bool *default_set)
f4grx's avatar
f4grx committed
  uint8_t buf;
  int ret;

  *default_set = false;

  /* Read the configuration enabled marker */
  ret = hn70ap_eeprom_read(0, &buf, 1);
  if(ret != OK)
    {
      return ret;
    }

  /* Not set? Set defaults */
  if( buf != HN70AP_CONFIG_INSTALLED)
    {
    /* Everything is set to FFFFh */
f4grx's avatar
f4grx committed
    *default_set = true;
    /* Clean ham call */
    hn70ap_eeconfig_setcall("call", "");

f4grx's avatar
f4grx committed
    buf = HN70AP_CONFIG_INSTALLED;
    hn70ap_eeprom_write(0, &buf, 1);
    }

  /* Done */

f4grx's avatar
f4grx committed
/*----------------------------------------------------------------------------*/
int hn70ap_eeprom_read(uint32_t addr, uint8_t *buf, uint32_t len)
{
  int fd;
  int ret = OK;
f4grx's avatar
f4grx committed

  fd = open(HN70AP_EECONFIG_DEVICE, O_RDONLY);
  if(fd < 0)
    {
      return ERROR;
    }
  if(lseek(fd, addr, SEEK_SET) != addr)
f4grx's avatar
f4grx committed
    {
      ret = ERROR;
      goto done;
f4grx's avatar
f4grx committed
    }
  if(read(fd, buf, len) != len)
    {
      ret = ERROR;
      goto done;
f4grx's avatar
f4grx committed
    }
f4grx's avatar
f4grx committed
  close(fd);
  return ret;
f4grx's avatar
f4grx committed
}

/*----------------------------------------------------------------------------*/
int hn70ap_eeprom_write(uint32_t addr, uint8_t *buf, uint32_t len)
{
  int fd;

  fd = open(HN70AP_EECONFIG_DEVICE, O_WRONLY);
  if(fd < 0)
    {
      return ERROR;
    }
  if(lseek(fd, addr, SEEK_SET) != addr)
f4grx's avatar
f4grx committed
    {
      close(fd);
      return ERROR;
    }
  if(write(fd, buf, len) != len)
    {
      close(fd);
      return ERROR;
    }
  close(fd);
  return OK;
}

f4grx's avatar
f4grx committed
/*----------------------------------------------------------------------------*/
int hn70ap_eeconfig_describe(int index, char *namebuf, int namebuflen, uint32_t *type)
{
  if(index < 0) return ERROR;
  if(index >= EECONFIG_PARAMS_COUNT) return ERROR;
  if(namebuf == NULL) return ERROR;

  strncpy(namebuf, eeconfig_params[index].name, namebuflen);
  if(type)
    {
      *type = eeconfig_params[index].type;
    }

  return OK;
}

f4grx's avatar
f4grx committed
/*----------------------------------------------------------------------------*/
f4grx's avatar
f4grx committed
int hn70ap_eeconfig_find(char *name)
f4grx's avatar
f4grx committed
{
  int i;
  for(i = 0; i < EECONFIG_PARAMS_COUNT; i++)
    {
      if(!strcmp(name, eeconfig_params[i].name))
        {
          return i;
        }
    }
  return -1;
}

f4grx's avatar
f4grx committed
/*----------------------------------------------------------------------------*/
int hn70ap_eeconfig_getbool(char *name, bool *value)
{
f4grx's avatar
f4grx committed
  int ret;
  uint8_t buf;
  int index = hn70ap_eeconfig_find(name);

f4grx's avatar
f4grx committed
  if(index < 0)
    {
      return ERROR;
    }
  if(eeconfig_params[index].type != EECONFIG_TYPE_BOOL)
    {
      return ERROR;
    }
  ret = hn70ap_eeprom_read(eeconfig_params[index].addr, &buf, 1);
  if(ret != OK)
    {
      return ERROR;
    }

  *value = (buf & eeconfig_params[index].mask) == eeconfig_params[index].mask;

  return OK;
f4grx's avatar
f4grx committed
}

/*----------------------------------------------------------------------------*/
f4grx's avatar
f4grx committed
int hn70ap_eeconfig_getip(char *name, in_addr_t *value)
f4grx's avatar
f4grx committed
{
f4grx's avatar
f4grx committed
  int ret;
  int index = hn70ap_eeconfig_find(name);

f4grx's avatar
f4grx committed
  if(index < 0)
    {
      return ERROR;
    }
  if(eeconfig_params[index].type != EECONFIG_TYPE_IP)
    {
      return ERROR;
    }

  ret = hn70ap_eeprom_read(eeconfig_params[index].addr, (uint8_t*)value, 4);
  if(ret != OK)
    {
      return ERROR;
    }

  return OK;
f4grx's avatar
f4grx committed
/*----------------------------------------------------------------------------*/
int hn70ap_eeconfig_getbyte(char *name, uint8_t *value)
{
  int ret;
  int index = hn70ap_eeconfig_find(name);

f4grx's avatar
f4grx committed
  if(index < 0)
    {
      return ERROR;
    }
  if(eeconfig_params[index].type != EECONFIG_TYPE_BYTE)
    {
      return ERROR;
    }
  ret = hn70ap_eeprom_read(eeconfig_params[index].addr, value, 1);
  if(ret != OK)
    {
      return ERROR;
    }

  return OK;
}

/*----------------------------------------------------------------------------*/
int hn70ap_eeconfig_getcall(char *name, char *value)
{
  int ret;
  int index = hn70ap_eeconfig_find(name);

f4grx's avatar
f4grx committed
  if(index < 0)
    {
      return ERROR;
    }
  if(eeconfig_params[index].type != EECONFIG_TYPE_CALL)
    {
      return ERROR;
    }
  ret = hn70ap_eeprom_read(eeconfig_params[index].addr, (uint8_t*)value, 8);
  if(ret != OK)
    {
      return ERROR;
    }

  return OK;
}

/*----------------------------------------------------------------------------*/
int hn70ap_eeconfig_setbool(char *name, bool value)
{
  int ret;
  uint8_t buf;
  int index = hn70ap_eeconfig_find(name);
f4grx's avatar
f4grx committed

  if(index < 0)
    {
      return ERROR;
    }
  if(eeconfig_params[index].type != EECONFIG_TYPE_BOOL)
    {
      return ERROR;
    }
  ret = hn70ap_eeprom_read(eeconfig_params[index].addr, &buf, 1);
  if(ret != OK)
    {
      return ERROR;
    }

  if(value)
    {
      buf |= eeconfig_params[index].mask;
    }
  else
    {
      buf &= ~eeconfig_params[index].mask;
    }

  ret = hn70ap_eeprom_write(eeconfig_params[index].addr, &buf, 1);
  if(ret != OK)
    {
      return ERROR;
    }

  return OK;
}

/*----------------------------------------------------------------------------*/
int hn70ap_eeconfig_setip(char *name, in_addr_t value)
{
  int ret;
  int index = hn70ap_eeconfig_find(name);

f4grx's avatar
f4grx committed
  if(index < 0)
    {
      return ERROR;
    }
  if(eeconfig_params[index].type != EECONFIG_TYPE_IP)
    {
      return ERROR;
    }
  ret = hn70ap_eeprom_write(eeconfig_params[index].addr, (uint8_t*)&value, 4);
  if(ret != OK)
    {
      return ERROR;
    }
  return OK;
}

/*----------------------------------------------------------------------------*/
int hn70ap_eeconfig_setbyte(char *name, uint8_t value)
{
  int ret;
  int index = hn70ap_eeconfig_find(name);

f4grx's avatar
f4grx committed
  if(index < 0)
    {
      return ERROR;
    }
  if(eeconfig_params[index].type != EECONFIG_TYPE_BYTE)
    {
      return ERROR;
    }
  ret = hn70ap_eeprom_write(eeconfig_params[index].addr, &value, 1);
  if(ret != OK)
    {
      return ERROR;
    }
  return OK;
}

/*----------------------------------------------------------------------------*/
int hn70ap_eeconfig_setcall(char *name, char *value)
{
  int ret;
  int i;
  int len;
  char call[8];
  int index = hn70ap_eeconfig_find(name);

f4grx's avatar
f4grx committed
  if(index < 0)
    {
      return ERROR;
    }
  if(eeconfig_params[index].type != EECONFIG_TYPE_CALL)
    {
      return ERROR;
    }

  /* Format callsign: only allow letters and numbers, turn into uppercase */
  len = strlen(value);
  if(len > 8)
    {
      len = 8;
    }
  for(i=0;i<len;i++)
    {
      if(value[i] >= 'a' && value[i] <= 'z')
        {
          call[i] = value[i] - 'a' + 'A';
        }
      else if(value[i] >= 'A' && value[i] <= 'Z')
        {
          call[i] = value[i];
        }
      else if(value[i] >= '0' && value[i] <= '9')
        {
          call[i] = value[i];
        }
      else
        {
          return ERROR;
        }
    }
  for(;i<8;i++)
    {
      call[i] = 0; /* Padding */
    }

  ret = hn70ap_eeprom_write(eeconfig_params[index].addr, (uint8_t*)call, 8);
f4grx's avatar
f4grx committed
  if(ret != OK)
    {
      return ERROR;
    }
  return OK;
}