/****************************************************************************
 * apps/system/hexed/src/hexmove.c
 * hexed move command
 *
 *   Copyright (c) 2011, B.ZaaR, All rights reserved.
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions are
 * met:
 *
 *   Redistributions of source code must retain the above copyright notice,
 *   this list of conditions and the following disclaimer.
 *
 *   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.
 *
 *   The names of contributors may not 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.
 *
 ****************************************************************************/

/****************************************************************************
 * Included Files
 ****************************************************************************/

#include <errno.h>
#include <stdint.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include "bfile.h"
#include "hexed.h"

/****************************************************************************
 * Private Functions
 ****************************************************************************/

/* Move len words from src to dest */

static int move(FAR struct command_s *cmd)
{
  bfmove(g_hexfile, cmd->opts.dest, cmd->opts.src, cmd->opts.bytes);
  return 0;
}

/* Move overwriting len words from src to dest */

static int moveover(FAR struct command_s *cmd)
{
  long cnt;
  long len;
  long off;

  bfmove(g_hexfile, cmd->opts.dest, cmd->opts.src, cmd->opts.bytes);

  /* Insert zeroes in chunks (keeps dest set) */

  if (cmd->opts.src < cmd->opts.dest)
    {
      memset(cmd->opts.buf, 0, sizeof(cmd->opts.buf));
      len = OPT_BUF_SZ;
      cnt = cmd->opts.bytes;
      off = cmd->opts.dest - cmd->opts.bytes;

      if (off < cmd->opts.src)
        {
          cnt = cmd->opts.dest - cmd->opts.src;
          off = cmd->opts.src;
        }

      for (; cnt; cnt -= len)
        {
          /* End of count? */

          if (cnt < len)
            {
              len = cnt;
            }

          /* Insert zeroes */

          bfwrite(g_hexfile, off, cmd->opts.buf, len);
          off += len;
        }

      /* Clip data */
    }
  else
    {
      bfclip(g_hexfile, cmd->opts.dest + cmd->opts.len, cmd->opts.bytes);
    }

  return 0;
}

/* Run the move command */

static int runmove(FAR struct command_s *cmd)
{
  /* Error: Source past EOF */

  if (cmd->opts.src > g_hexfile->size)
    {
      return -EINVAL;
    }

  switch (cmd->id)
    {
    case CMD_MOVE:
      return move(cmd);

    case CMD_MOVE_OVER:
      return moveover(cmd);

    default:
      return -1;
    }
}

/* Set the move command */

static int setmove(FAR struct command_s *cmd, int optc, char *opt)
{
  FAR char *s;
  uint64_t v;

  /* Set defaults */

  if (optc == 0)
    {
      cmd->flags |= CMD_FL_QUIT;
    }

  /* NULL option */

  if (opt == NULL)
    {
      return -EFAULT;
    }

  v = strtoll(opt, &s, 0x10);

  /* No value set in option */

  if (s == opt)
    {
      return -EINVAL;
    }

  /* Set options */

  switch (optc)
    {
    case 0:

      /* Set source */

      cmd->opts.src = v;
      optc++;
      break;

    case 1:

      /* Set destination */

      cmd->opts.dest = v;
      optc++;
      break;

    case 2:

      /* Set length */

      cmd->opts.len = v;
      cmd->opts.bytes = cmd->opts.len * cmd->opts.word;
      optc = 0;
      break;

    default:

      /* Too many options specified */

      return -E2BIG;
    }

  return optc;
}

/****************************************************************************
 * Public Functions
 ****************************************************************************/

/* Move command */

int hexmove(FAR struct command_s *cmd, int optc, char *opt)
{
  /* Invalid command */

  if (cmd == NULL || (cmd->id != CMD_MOVE && cmd->id != CMD_MOVE_OVER))
    {
      return -EINVAL;
    }

  /* Ret/run move */

  if (optc >= 0)
    {
      optc = setmove(cmd, optc, opt);
    }
  else
    {
      optc = runmove(cmd);
    }

  return optc;
}