/*  This file is part of the program psim.

    Copyright (C) 1994-1996, Andrew Cagney <cagney@highland.com.au>

    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; either version 2 of the License, or
    (at your option) any later version.

    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
    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.
 
    */


#ifndef _OS_EMUL_C_
#define _OS_EMUL_C_

#include "cpu.h"
#include "idecode.h"
#include "os_emul.h"

#include "emul_generic.h"
#include "emul_netbsd.h"
#include "emul_unix.h"
#include "emul_chirp.h"
#include "emul_bugapi.h"

static const os_emul *(os_emulations[]) = {
  &emul_chirp,
  &emul_bugapi,
  &emul_netbsd,
  &emul_solaris,
  &emul_linux,
  0
};


INLINE_OS_EMUL\
(os_emul *)
os_emul_create(const char *file_name,
	       device *root)
{
  const char *emulation_name = NULL;
  bfd *image;
  os_emul *chosen_emulation = NULL;

  bfd_init(); /* would never hurt */

  /* open the file */
  image = bfd_openr(file_name, NULL);
  if (image == NULL) {
    bfd_perror(file_name);
    error("nothing loaded\n");
  }

  /* check it is an executable */
  if (!bfd_check_format(image, bfd_object)) {
    TRACE(trace_tbd,
	  ("FIXME - should check more than just bfd_check_format\n"));
    TRACE(trace_os_emul,
	  ("%s not an executable, assumeing a device file\n", file_name));
    bfd_close(image);
    image = NULL;
  }

  /* if a device file, load that before trying the emulations on */
  if (image == NULL) {
    psim_merge_device_file(root, file_name);
  }

  /* see if the device tree already specifies the required emulation */
  if (tree_find_property(root, "/openprom/options/os-emul") != NULL)
    emulation_name =
      tree_find_string_property(root, "/openprom/options/os-emul");
  else
    emulation_name = NULL;

  /* go through each emulation to see if they reconize it. FIXME -
     should have some sort of imported table from a separate file */
  {
    os_emul_data *emul_data;
    const os_emul **possible_emulation;
    chosen_emulation = NULL;
    for (possible_emulation = os_emulations, emul_data = NULL;
	 *possible_emulation != NULL && emul_data == NULL;
	 possible_emulation++) {
      emul_data = (*possible_emulation)->create(root,
						image,
						emulation_name);
      if (emul_data != NULL) {
	chosen_emulation = ZALLOC(os_emul);
	*chosen_emulation = **possible_emulation;
	chosen_emulation->data = emul_data;
      }
    }
  }

  /* clean up */
  if (image != NULL)
    bfd_close(image);
  return chosen_emulation;
}

INLINE_OS_EMUL\
(void)
os_emul_init(os_emul *emulation,
	     int nr_cpus)
{
  if (emulation != (os_emul*)0)
    emulation->init(emulation->data, nr_cpus);
}

INLINE_OS_EMUL\
(void)
os_emul_system_call(cpu *processor,
		    unsigned_word cia)
{
  os_emul *emulation = cpu_os_emulation(processor);
  if (emulation != (os_emul*)0 && emulation->system_call != 0)
    emulation->system_call(processor, cia, emulation->data);
  else
    error("System call emulation not available\n");
}

INLINE_OS_EMUL\
(int)
os_emul_instruction_call(cpu *processor,
			 unsigned_word cia,
			 unsigned_word ra)
{
  os_emul *emulation = cpu_os_emulation(processor);
  if (emulation != (os_emul*)0 && emulation->instruction_call != 0)
    return emulation->instruction_call(processor, cia, ra, emulation->data);
  else
    return 0;
}


#endif /* _OS_EMUL_C_ */