2010-06-29 04:50:43 +02:00
|
|
|
/*
|
2020-04-26 12:16:37 +02:00
|
|
|
* QEMU fuloong 2e mini pc support
|
2010-06-29 04:50:43 +02:00
|
|
|
*
|
|
|
|
* Copyright (c) 2008 yajin (yajin@vm-kernel.org)
|
|
|
|
* Copyright (c) 2009 chenming (chenming@rdc.faw.com.cn)
|
|
|
|
* Copyright (c) 2010 Huacai Chen (zltjiangshi@gmail.com)
|
|
|
|
* This code is licensed under the GNU GPL v2.
|
2012-01-13 17:44:23 +01:00
|
|
|
*
|
|
|
|
* Contributions after 2012-01-13 are licensed under the terms of the
|
|
|
|
* GNU GPL, version 2 or (at your option) any later version.
|
2010-06-29 04:50:43 +02:00
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
2020-04-26 12:16:37 +02:00
|
|
|
* Fuloong 2e mini pc is based on ICT/ST Loongson 2e CPU (MIPS III like, 800MHz)
|
|
|
|
* https://www.linux-mips.org/wiki/Fuloong_2E
|
2010-06-29 04:50:43 +02:00
|
|
|
*
|
|
|
|
* Loongson 2e user manual:
|
|
|
|
* http://www.loongsondeveloper.com/doc/Loongson2EUserGuide.pdf
|
|
|
|
*/
|
|
|
|
|
2016-01-18 18:35:00 +01:00
|
|
|
#include "qemu/osdep.h"
|
2019-05-23 16:35:08 +02:00
|
|
|
#include "qemu-common.h"
|
2018-06-25 14:42:22 +02:00
|
|
|
#include "qemu/units.h"
|
include/qemu/osdep.h: Don't include qapi/error.h
Commit 57cb38b included qapi/error.h into qemu/osdep.h to get the
Error typedef. Since then, we've moved to include qemu/osdep.h
everywhere. Its file comment explains: "To avoid getting into
possible circular include dependencies, this file should not include
any other QEMU headers, with the exceptions of config-host.h,
compiler.h, os-posix.h and os-win32.h, all of which are doing a
similar job to this file and are under similar constraints."
qapi/error.h doesn't do a similar job, and it doesn't adhere to
similar constraints: it includes qapi-types.h. That's in excess of
100KiB of crap most .c files don't actually need.
Add the typedef to qemu/typedefs.h, and include that instead of
qapi/error.h. Include qapi/error.h in .c files that need it and don't
get it now. Include qapi-types.h in qom/object.h for uint16List.
Update scripts/clean-includes accordingly. Update it further to match
reality: replace config.h by config-target.h, add sysemu/os-posix.h,
sysemu/os-win32.h. Update the list of includes in the qemu/osdep.h
comment quoted above similarly.
This reduces the number of objects depending on qapi/error.h from "all
of them" to less than a third. Unfortunately, the number depending on
qapi-types.h shrinks only a little. More work is needed for that one.
Signed-off-by: Markus Armbruster <armbru@redhat.com>
[Fix compilation without the spice devel packages. - Paolo]
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2016-03-14 09:01:28 +01:00
|
|
|
#include "qapi/error.h"
|
2019-02-21 13:29:15 +01:00
|
|
|
#include "cpu.h"
|
2019-12-12 17:15:43 +01:00
|
|
|
#include "hw/intc/i8259.h"
|
2018-03-08 23:39:23 +01:00
|
|
|
#include "hw/dma/i8257.h"
|
2018-03-08 23:39:40 +01:00
|
|
|
#include "hw/isa/superio.h"
|
2012-10-24 08:43:34 +02:00
|
|
|
#include "net/net.h"
|
2013-02-04 15:40:22 +01:00
|
|
|
#include "hw/boards.h"
|
2018-11-14 01:31:27 +01:00
|
|
|
#include "hw/i2c/smbus_eeprom.h"
|
2013-02-05 17:06:20 +01:00
|
|
|
#include "hw/block/flash.h"
|
|
|
|
#include "hw/mips/mips.h"
|
|
|
|
#include "hw/mips/cpudevs.h"
|
2013-02-04 15:40:22 +01:00
|
|
|
#include "hw/pci/pci.h"
|
2012-12-17 18:20:00 +01:00
|
|
|
#include "qemu/log.h"
|
2013-02-04 15:40:22 +01:00
|
|
|
#include "hw/loader.h"
|
2020-03-13 09:24:39 +01:00
|
|
|
#include "hw/ide/pci.h"
|
2010-06-29 04:50:43 +02:00
|
|
|
#include "elf.h"
|
2013-02-05 17:06:20 +01:00
|
|
|
#include "hw/isa/vt82c686.h"
|
2019-10-04 01:03:53 +02:00
|
|
|
#include "hw/rtc/mc146818rtc.h"
|
2013-02-05 17:06:20 +01:00
|
|
|
#include "hw/timer/i8254.h"
|
2012-12-17 18:19:49 +01:00
|
|
|
#include "exec/address-spaces.h"
|
2013-07-29 16:05:30 +02:00
|
|
|
#include "sysemu/qtest.h"
|
2019-08-12 07:23:38 +02:00
|
|
|
#include "sysemu/reset.h"
|
2013-08-03 16:03:18 +02:00
|
|
|
#include "qemu/error-report.h"
|
2010-06-29 04:50:43 +02:00
|
|
|
|
2020-04-26 12:16:37 +02:00
|
|
|
#define DEBUG_FULOONG2E_INIT
|
2010-06-29 04:50:43 +02:00
|
|
|
|
2019-08-19 14:07:52 +02:00
|
|
|
#define ENVP_ADDR 0x80002000l
|
|
|
|
#define ENVP_NB_ENTRIES 16
|
|
|
|
#define ENVP_ENTRY_SIZE 256
|
2010-06-29 04:50:43 +02:00
|
|
|
|
2020-04-26 12:16:37 +02:00
|
|
|
/* Fuloong 2e has a 512k flash: Winbond W39L040AP70Z */
|
2019-08-19 14:07:52 +02:00
|
|
|
#define BIOS_SIZE (512 * KiB)
|
|
|
|
#define MAX_IDE_BUS 2
|
2010-06-29 04:50:43 +02:00
|
|
|
|
|
|
|
/*
|
|
|
|
* PMON is not part of qemu and released with BSD license, anyone
|
|
|
|
* who want to build a pmon binary please first git-clone the source
|
|
|
|
* from the git repository at:
|
|
|
|
* http://www.loongson.cn/support/git/pmon
|
|
|
|
* Then follow the "Compile Guide" available at:
|
|
|
|
* http://dev.lemote.com/code/pmon
|
|
|
|
*
|
|
|
|
* Notes:
|
|
|
|
* 1, don't use the source at http://dev.lemote.com/http_git/pmon.git
|
|
|
|
* 2, use "Bonito2edev" to replace "dir_corresponding_to_your_target_hardware"
|
|
|
|
* in the "Compile Guide".
|
|
|
|
*/
|
2020-04-26 12:16:37 +02:00
|
|
|
#define FULOONG_BIOSNAME "pmon_2e.bin"
|
2010-06-29 04:50:43 +02:00
|
|
|
|
2020-04-26 12:16:37 +02:00
|
|
|
/* PCI SLOT in Fuloong 2e */
|
|
|
|
#define FULOONG2E_VIA_SLOT 5
|
|
|
|
#define FULOONG2E_ATI_SLOT 6
|
|
|
|
#define FULOONG2E_RTL8139_SLOT 7
|
2010-06-29 04:50:43 +02:00
|
|
|
|
|
|
|
static struct _loaderparams {
|
|
|
|
int ram_size;
|
|
|
|
const char *kernel_filename;
|
|
|
|
const char *kernel_cmdline;
|
|
|
|
const char *initrd_filename;
|
|
|
|
} loaderparams;
|
|
|
|
|
2019-08-19 14:07:52 +02:00
|
|
|
static void GCC_FMT_ATTR(3, 4) prom_set(uint32_t *prom_buf, int index,
|
2010-09-23 21:28:05 +02:00
|
|
|
const char *string, ...)
|
2010-06-29 04:50:43 +02:00
|
|
|
{
|
|
|
|
va_list ap;
|
|
|
|
int32_t table_addr;
|
|
|
|
|
2019-08-19 14:07:52 +02:00
|
|
|
if (index >= ENVP_NB_ENTRIES) {
|
2010-06-29 04:50:43 +02:00
|
|
|
return;
|
2019-08-19 14:07:52 +02:00
|
|
|
}
|
2010-06-29 04:50:43 +02:00
|
|
|
|
|
|
|
if (string == NULL) {
|
|
|
|
prom_buf[index] = 0;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
table_addr = sizeof(int32_t) * ENVP_NB_ENTRIES + index * ENVP_ENTRY_SIZE;
|
|
|
|
prom_buf[index] = tswap32(ENVP_ADDR + table_addr);
|
|
|
|
|
|
|
|
va_start(ap, string);
|
|
|
|
vsnprintf((char *)prom_buf + table_addr, ENVP_ENTRY_SIZE, string, ap);
|
|
|
|
va_end(ap);
|
|
|
|
}
|
|
|
|
|
2019-08-19 14:07:52 +02:00
|
|
|
static int64_t load_kernel(CPUMIPSState *env)
|
2010-06-29 04:50:43 +02:00
|
|
|
{
|
2018-09-13 12:07:13 +02:00
|
|
|
int64_t kernel_entry, kernel_low, kernel_high, initrd_size;
|
2010-06-29 04:50:43 +02:00
|
|
|
int index = 0;
|
2018-09-13 12:07:13 +02:00
|
|
|
long kernel_size;
|
2010-06-29 04:50:43 +02:00
|
|
|
ram_addr_t initrd_offset;
|
|
|
|
uint32_t *prom_buf;
|
|
|
|
long prom_size;
|
|
|
|
|
2019-01-15 13:18:03 +01:00
|
|
|
kernel_size = load_elf(loaderparams.kernel_filename, NULL,
|
|
|
|
cpu_mips_kseg0_to_phys, NULL,
|
|
|
|
(uint64_t *)&kernel_entry,
|
2017-07-27 01:56:13 +02:00
|
|
|
(uint64_t *)&kernel_low, (uint64_t *)&kernel_high,
|
2020-01-26 23:55:04 +01:00
|
|
|
NULL, 0, EM_MIPS, 1, 0);
|
2017-07-27 01:56:13 +02:00
|
|
|
if (kernel_size < 0) {
|
hw/mips: Replace fprintf(stderr, "*\n" with error_report()
Replace a large number of the fprintf(stderr, "*\n" calls with
error_report(). The functions were renamed with these commands and then
compiler issues where manually fixed.
find ./* -type f -exec sed -i \
'N;N;N;N;N;N;N;N;N;N;N;N; {s|fprintf(stderr, "\(.*\)\\n"\(.*\));|error_report("\1"\2);|Ig}' \
{} +
find ./* -type f -exec sed -i \
'N;N;N;N;N;N;N;N;N;N;N; {s|fprintf(stderr, "\(.*\)\\n"\(.*\));|error_report("\1"\2);|Ig}' \
{} +
find ./* -type f -exec sed -i \
'N;N;N;N;N;N;N;N;N; {s|fprintf(stderr, "\(.*\)\\n"\(.*\));|error_report("\1"\2);|Ig}' \
{} +
find ./* -type f -exec sed -i \
'N;N;N;N;N;N;N;N; {s|fprintf(stderr, "\(.*\)\\n"\(.*\));|error_report("\1"\2);|Ig}' \
{} +
find ./* -type f -exec sed -i \
'N;N;N;N;N;N;N; {s|fprintf(stderr, "\(.*\)\\n"\(.*\));|error_report("\1"\2);|Ig}' \
{} +
find ./* -type f -exec sed -i \
'N;N;N;N;N;N; {s|fprintf(stderr, "\(.*\)\\n"\(.*\));|error_report("\1"\2);|Ig}' \
{} +
find ./* -type f -exec sed -i \
'N;N;N;N;N; {s|fprintf(stderr, "\(.*\)\\n"\(.*\));|error_report("\1"\2);|Ig}' \
{} +
find ./* -type f -exec sed -i \
'N;N;N;N; {s|fprintf(stderr, "\(.*\)\\n"\(.*\));|error_report("\1"\2);|Ig}' \
{} +
find ./* -type f -exec sed -i \
'N;N;N; {s|fprintf(stderr, "\(.*\)\\n"\(.*\));|error_report("\1"\2);|Ig}' \
{} +
find ./* -type f -exec sed -i \
'N;N; {s|fprintf(stderr, "\(.*\)\\n"\(.*\));|error_report("\1"\2);|Ig}' \
{} +
find ./* -type f -exec sed -i \
'N; {s|fprintf(stderr, "\(.*\)\\n"\(.*\));|error_report("\1"\2);|Ig}' \
{} +
Some lines where then manually tweaked to pass checkpatch.
Signed-off-by: Alistair Francis <alistair.francis@xilinx.com>
Cc: Paul Burton <paul.burton@imgtec.com>
Cc: Aurelien Jarno <aurelien@aurel32.net>
Cc: Yongbok Kim <yongbok.kim@imgtec.com>
Cc: "Hervé Poussineau" <hpoussin@reactos.org>
Conversions that aren't followed by exit() dropped, because they might
be inappropriate.
Signed-off-by: Markus Armbruster <armbru@redhat.com>
Reviewed-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
Message-Id: <20180203084315.20497-6-armbru@redhat.com>
2018-02-03 09:43:06 +01:00
|
|
|
error_report("could not load kernel '%s': %s",
|
2017-07-27 01:56:13 +02:00
|
|
|
loaderparams.kernel_filename,
|
|
|
|
load_elf_strerror(kernel_size));
|
2010-06-29 04:50:43 +02:00
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* load initrd */
|
|
|
|
initrd_size = 0;
|
|
|
|
initrd_offset = 0;
|
|
|
|
if (loaderparams.initrd_filename) {
|
2019-08-19 14:07:52 +02:00
|
|
|
initrd_size = get_image_size(loaderparams.initrd_filename);
|
2010-06-29 04:50:43 +02:00
|
|
|
if (initrd_size > 0) {
|
2019-08-19 14:07:52 +02:00
|
|
|
initrd_offset = (kernel_high + ~INITRD_PAGE_MASK) &
|
|
|
|
INITRD_PAGE_MASK;
|
2010-06-29 04:50:43 +02:00
|
|
|
if (initrd_offset + initrd_size > ram_size) {
|
hw/mips: Replace fprintf(stderr, "*\n" with error_report()
Replace a large number of the fprintf(stderr, "*\n" calls with
error_report(). The functions were renamed with these commands and then
compiler issues where manually fixed.
find ./* -type f -exec sed -i \
'N;N;N;N;N;N;N;N;N;N;N;N; {s|fprintf(stderr, "\(.*\)\\n"\(.*\));|error_report("\1"\2);|Ig}' \
{} +
find ./* -type f -exec sed -i \
'N;N;N;N;N;N;N;N;N;N;N; {s|fprintf(stderr, "\(.*\)\\n"\(.*\));|error_report("\1"\2);|Ig}' \
{} +
find ./* -type f -exec sed -i \
'N;N;N;N;N;N;N;N;N; {s|fprintf(stderr, "\(.*\)\\n"\(.*\));|error_report("\1"\2);|Ig}' \
{} +
find ./* -type f -exec sed -i \
'N;N;N;N;N;N;N;N; {s|fprintf(stderr, "\(.*\)\\n"\(.*\));|error_report("\1"\2);|Ig}' \
{} +
find ./* -type f -exec sed -i \
'N;N;N;N;N;N;N; {s|fprintf(stderr, "\(.*\)\\n"\(.*\));|error_report("\1"\2);|Ig}' \
{} +
find ./* -type f -exec sed -i \
'N;N;N;N;N;N; {s|fprintf(stderr, "\(.*\)\\n"\(.*\));|error_report("\1"\2);|Ig}' \
{} +
find ./* -type f -exec sed -i \
'N;N;N;N;N; {s|fprintf(stderr, "\(.*\)\\n"\(.*\));|error_report("\1"\2);|Ig}' \
{} +
find ./* -type f -exec sed -i \
'N;N;N;N; {s|fprintf(stderr, "\(.*\)\\n"\(.*\));|error_report("\1"\2);|Ig}' \
{} +
find ./* -type f -exec sed -i \
'N;N;N; {s|fprintf(stderr, "\(.*\)\\n"\(.*\));|error_report("\1"\2);|Ig}' \
{} +
find ./* -type f -exec sed -i \
'N;N; {s|fprintf(stderr, "\(.*\)\\n"\(.*\));|error_report("\1"\2);|Ig}' \
{} +
find ./* -type f -exec sed -i \
'N; {s|fprintf(stderr, "\(.*\)\\n"\(.*\));|error_report("\1"\2);|Ig}' \
{} +
Some lines where then manually tweaked to pass checkpatch.
Signed-off-by: Alistair Francis <alistair.francis@xilinx.com>
Cc: Paul Burton <paul.burton@imgtec.com>
Cc: Aurelien Jarno <aurelien@aurel32.net>
Cc: Yongbok Kim <yongbok.kim@imgtec.com>
Cc: "Hervé Poussineau" <hpoussin@reactos.org>
Conversions that aren't followed by exit() dropped, because they might
be inappropriate.
Signed-off-by: Markus Armbruster <armbru@redhat.com>
Reviewed-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
Message-Id: <20180203084315.20497-6-armbru@redhat.com>
2018-02-03 09:43:06 +01:00
|
|
|
error_report("memory too small for initial ram disk '%s'",
|
|
|
|
loaderparams.initrd_filename);
|
2010-06-29 04:50:43 +02:00
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
initrd_size = load_image_targphys(loaderparams.initrd_filename,
|
2019-08-19 14:07:52 +02:00
|
|
|
initrd_offset,
|
|
|
|
ram_size - initrd_offset);
|
2010-06-29 04:50:43 +02:00
|
|
|
}
|
|
|
|
if (initrd_size == (target_ulong) -1) {
|
hw/mips: Replace fprintf(stderr, "*\n" with error_report()
Replace a large number of the fprintf(stderr, "*\n" calls with
error_report(). The functions were renamed with these commands and then
compiler issues where manually fixed.
find ./* -type f -exec sed -i \
'N;N;N;N;N;N;N;N;N;N;N;N; {s|fprintf(stderr, "\(.*\)\\n"\(.*\));|error_report("\1"\2);|Ig}' \
{} +
find ./* -type f -exec sed -i \
'N;N;N;N;N;N;N;N;N;N;N; {s|fprintf(stderr, "\(.*\)\\n"\(.*\));|error_report("\1"\2);|Ig}' \
{} +
find ./* -type f -exec sed -i \
'N;N;N;N;N;N;N;N;N; {s|fprintf(stderr, "\(.*\)\\n"\(.*\));|error_report("\1"\2);|Ig}' \
{} +
find ./* -type f -exec sed -i \
'N;N;N;N;N;N;N;N; {s|fprintf(stderr, "\(.*\)\\n"\(.*\));|error_report("\1"\2);|Ig}' \
{} +
find ./* -type f -exec sed -i \
'N;N;N;N;N;N;N; {s|fprintf(stderr, "\(.*\)\\n"\(.*\));|error_report("\1"\2);|Ig}' \
{} +
find ./* -type f -exec sed -i \
'N;N;N;N;N;N; {s|fprintf(stderr, "\(.*\)\\n"\(.*\));|error_report("\1"\2);|Ig}' \
{} +
find ./* -type f -exec sed -i \
'N;N;N;N;N; {s|fprintf(stderr, "\(.*\)\\n"\(.*\));|error_report("\1"\2);|Ig}' \
{} +
find ./* -type f -exec sed -i \
'N;N;N;N; {s|fprintf(stderr, "\(.*\)\\n"\(.*\));|error_report("\1"\2);|Ig}' \
{} +
find ./* -type f -exec sed -i \
'N;N;N; {s|fprintf(stderr, "\(.*\)\\n"\(.*\));|error_report("\1"\2);|Ig}' \
{} +
find ./* -type f -exec sed -i \
'N;N; {s|fprintf(stderr, "\(.*\)\\n"\(.*\));|error_report("\1"\2);|Ig}' \
{} +
find ./* -type f -exec sed -i \
'N; {s|fprintf(stderr, "\(.*\)\\n"\(.*\));|error_report("\1"\2);|Ig}' \
{} +
Some lines where then manually tweaked to pass checkpatch.
Signed-off-by: Alistair Francis <alistair.francis@xilinx.com>
Cc: Paul Burton <paul.burton@imgtec.com>
Cc: Aurelien Jarno <aurelien@aurel32.net>
Cc: Yongbok Kim <yongbok.kim@imgtec.com>
Cc: "Hervé Poussineau" <hpoussin@reactos.org>
Conversions that aren't followed by exit() dropped, because they might
be inappropriate.
Signed-off-by: Markus Armbruster <armbru@redhat.com>
Reviewed-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
Message-Id: <20180203084315.20497-6-armbru@redhat.com>
2018-02-03 09:43:06 +01:00
|
|
|
error_report("could not load initial ram disk '%s'",
|
|
|
|
loaderparams.initrd_filename);
|
2010-06-29 04:50:43 +02:00
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Setup prom parameters. */
|
|
|
|
prom_size = ENVP_NB_ENTRIES * (sizeof(int32_t) + ENVP_ENTRY_SIZE);
|
2011-08-21 05:09:37 +02:00
|
|
|
prom_buf = g_malloc(prom_size);
|
2010-06-29 04:50:43 +02:00
|
|
|
|
2010-09-20 22:18:00 +02:00
|
|
|
prom_set(prom_buf, index++, "%s", loaderparams.kernel_filename);
|
2010-06-29 04:50:43 +02:00
|
|
|
if (initrd_size > 0) {
|
2019-08-19 14:07:52 +02:00
|
|
|
prom_set(prom_buf, index++,
|
|
|
|
"rd_start=0x%" PRIx64 " rd_size=%" PRId64 " %s",
|
|
|
|
cpu_mips_phys_to_kseg0(NULL, initrd_offset),
|
|
|
|
initrd_size, loaderparams.kernel_cmdline);
|
2010-06-29 04:50:43 +02:00
|
|
|
} else {
|
2010-09-20 22:18:00 +02:00
|
|
|
prom_set(prom_buf, index++, "%s", loaderparams.kernel_cmdline);
|
2010-06-29 04:50:43 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Setup minimum environment variables */
|
|
|
|
prom_set(prom_buf, index++, "busclock=33000000");
|
|
|
|
prom_set(prom_buf, index++, "cpuclock=100000000");
|
2018-06-25 14:42:22 +02:00
|
|
|
prom_set(prom_buf, index++, "memsize=%"PRIi64, loaderparams.ram_size / MiB);
|
2010-06-29 04:50:43 +02:00
|
|
|
prom_set(prom_buf, index++, "modetty0=38400n8r");
|
|
|
|
prom_set(prom_buf, index++, NULL);
|
|
|
|
|
|
|
|
rom_add_blob_fixed("prom", prom_buf, prom_size,
|
|
|
|
cpu_mips_kseg0_to_phys(NULL, ENVP_ADDR));
|
|
|
|
|
2015-04-28 11:11:02 +02:00
|
|
|
g_free(prom_buf);
|
2010-06-29 04:50:43 +02:00
|
|
|
return kernel_entry;
|
|
|
|
}
|
|
|
|
|
2019-08-19 14:07:52 +02:00
|
|
|
static void write_bootloader(CPUMIPSState *env, uint8_t *base,
|
|
|
|
int64_t kernel_addr)
|
2010-06-29 04:50:43 +02:00
|
|
|
{
|
|
|
|
uint32_t *p;
|
|
|
|
|
|
|
|
/* Small bootloader */
|
2019-08-19 14:07:52 +02:00
|
|
|
p = (uint32_t *)base;
|
2010-06-29 04:50:43 +02:00
|
|
|
|
2019-08-19 14:07:52 +02:00
|
|
|
/* j 0x1fc00040 */
|
|
|
|
stl_p(p++, 0x0bf00010);
|
|
|
|
/* nop */
|
|
|
|
stl_p(p++, 0x00000000);
|
2010-06-29 04:50:43 +02:00
|
|
|
|
|
|
|
/* Second part of the bootloader */
|
2019-08-19 14:07:52 +02:00
|
|
|
p = (uint32_t *)(base + 0x040);
|
|
|
|
|
|
|
|
/* lui a0, 0 */
|
|
|
|
stl_p(p++, 0x3c040000);
|
|
|
|
/* ori a0, a0, 2 */
|
|
|
|
stl_p(p++, 0x34840002);
|
|
|
|
/* lui a1, high(ENVP_ADDR) */
|
|
|
|
stl_p(p++, 0x3c050000 | ((ENVP_ADDR >> 16) & 0xffff));
|
|
|
|
/* ori a1, a0, low(ENVP_ADDR) */
|
|
|
|
stl_p(p++, 0x34a50000 | (ENVP_ADDR & 0xffff));
|
|
|
|
/* lui a2, high(ENVP_ADDR + 8) */
|
|
|
|
stl_p(p++, 0x3c060000 | (((ENVP_ADDR + 8) >> 16) & 0xffff));
|
|
|
|
/* ori a2, a2, low(ENVP_ADDR + 8) */
|
|
|
|
stl_p(p++, 0x34c60000 | ((ENVP_ADDR + 8) & 0xffff));
|
|
|
|
/* lui a3, high(env->ram_size) */
|
|
|
|
stl_p(p++, 0x3c070000 | (loaderparams.ram_size >> 16));
|
|
|
|
/* ori a3, a3, low(env->ram_size) */
|
|
|
|
stl_p(p++, 0x34e70000 | (loaderparams.ram_size & 0xffff));
|
|
|
|
/* lui ra, high(kernel_addr) */
|
|
|
|
stl_p(p++, 0x3c1f0000 | ((kernel_addr >> 16) & 0xffff));
|
|
|
|
/* ori ra, ra, low(kernel_addr) */
|
|
|
|
stl_p(p++, 0x37ff0000 | (kernel_addr & 0xffff));
|
|
|
|
/* jr ra */
|
|
|
|
stl_p(p++, 0x03e00008);
|
|
|
|
/* nop */
|
|
|
|
stl_p(p++, 0x00000000);
|
2010-06-29 04:50:43 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static void main_cpu_reset(void *opaque)
|
|
|
|
{
|
2012-05-05 13:56:22 +02:00
|
|
|
MIPSCPU *cpu = opaque;
|
|
|
|
CPUMIPSState *env = &cpu->env;
|
2010-06-29 04:50:43 +02:00
|
|
|
|
2012-05-05 13:56:22 +02:00
|
|
|
cpu_reset(CPU(cpu));
|
2010-06-29 04:50:43 +02:00
|
|
|
/* TODO: 2E reset stuff */
|
|
|
|
if (loaderparams.kernel_filename) {
|
|
|
|
env->CP0_Status &= ~((1 << CP0St_BEV) | (1 << CP0St_ERL));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-03-08 23:39:38 +01:00
|
|
|
static void vt82c686b_southbridge_init(PCIBus *pci_bus, int slot, qemu_irq intc,
|
|
|
|
I2CBus **i2c_bus, ISABus **p_isa_bus)
|
2010-06-29 04:50:43 +02:00
|
|
|
{
|
2018-03-08 23:39:38 +01:00
|
|
|
qemu_irq *i8259;
|
|
|
|
ISABus *isa_bus;
|
2020-03-13 09:24:39 +01:00
|
|
|
PCIDevice *dev;
|
2018-03-08 23:39:38 +01:00
|
|
|
|
2018-03-08 23:39:39 +01:00
|
|
|
isa_bus = vt82c686b_isa_init(pci_bus, PCI_DEVFN(slot, 0));
|
2018-03-08 23:39:38 +01:00
|
|
|
if (!isa_bus) {
|
|
|
|
fprintf(stderr, "vt82c686b_init error\n");
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
*p_isa_bus = isa_bus;
|
|
|
|
/* Interrupt controller */
|
|
|
|
/* The 8259 -> IP5 */
|
|
|
|
i8259 = i8259_init(isa_bus, intc);
|
|
|
|
isa_bus_irqs(isa_bus, i8259);
|
|
|
|
/* init other devices */
|
|
|
|
i8254_pit_init(isa_bus, 0x40, 0, NULL);
|
|
|
|
i8257_dma_init(isa_bus, 0);
|
2018-03-08 23:39:40 +01:00
|
|
|
/* Super I/O */
|
|
|
|
isa_create_simple(isa_bus, TYPE_VT82C686B_SUPERIO);
|
2018-03-08 23:39:38 +01:00
|
|
|
|
2020-03-13 09:24:39 +01:00
|
|
|
dev = pci_create_simple(pci_bus, PCI_DEVFN(slot, 1), "via-ide");
|
2020-03-17 16:05:37 +01:00
|
|
|
pci_ide_create_devs(dev);
|
2018-03-08 23:39:38 +01:00
|
|
|
|
|
|
|
pci_create_simple(pci_bus, PCI_DEVFN(slot, 2), "vt82c686b-usb-uhci");
|
|
|
|
pci_create_simple(pci_bus, PCI_DEVFN(slot, 3), "vt82c686b-usb-uhci");
|
|
|
|
|
|
|
|
*i2c_bus = vt82c686b_pm_init(pci_bus, PCI_DEVFN(slot, 4), 0xeee1, NULL);
|
|
|
|
|
|
|
|
/* Audio support */
|
|
|
|
vt82c686b_ac97_init(pci_bus, PCI_DEVFN(slot, 5));
|
|
|
|
vt82c686b_mc97_init(pci_bus, PCI_DEVFN(slot, 6));
|
2010-06-29 04:50:43 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Network support */
|
2019-08-19 14:07:52 +02:00
|
|
|
static void network_init(PCIBus *pci_bus)
|
2010-06-29 04:50:43 +02:00
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
2019-08-19 14:07:52 +02:00
|
|
|
for (i = 0; i < nb_nics; i++) {
|
2010-06-29 04:50:43 +02:00
|
|
|
NICInfo *nd = &nd_table[i];
|
|
|
|
const char *default_devaddr = NULL;
|
|
|
|
|
|
|
|
if (i == 0 && (!nd->model || strcmp(nd->model, "rtl8139") == 0)) {
|
2020-04-26 12:16:37 +02:00
|
|
|
/* The Fuloong board has a RTL8139 card using PCI SLOT 7 */
|
2010-06-29 04:50:43 +02:00
|
|
|
default_devaddr = "07";
|
|
|
|
}
|
|
|
|
|
2013-06-06 10:48:51 +02:00
|
|
|
pci_nic_init_nofail(nd, pci_bus, "rtl8139", default_devaddr);
|
2010-06-29 04:50:43 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-04-26 12:16:37 +02:00
|
|
|
static void mips_fuloong2e_init(MachineState *machine)
|
2010-06-29 04:50:43 +02:00
|
|
|
{
|
2014-05-07 16:42:57 +02:00
|
|
|
const char *kernel_filename = machine->kernel_filename;
|
|
|
|
const char *kernel_cmdline = machine->kernel_cmdline;
|
|
|
|
const char *initrd_filename = machine->initrd_filename;
|
2010-06-29 04:50:43 +02:00
|
|
|
char *filename;
|
2011-08-08 20:46:13 +02:00
|
|
|
MemoryRegion *address_space_mem = get_system_memory();
|
|
|
|
MemoryRegion *bios = g_new(MemoryRegion, 1);
|
2010-09-18 07:53:14 +02:00
|
|
|
long bios_size;
|
2019-02-21 13:34:07 +01:00
|
|
|
uint8_t *spd_data;
|
2010-06-29 04:50:43 +02:00
|
|
|
int64_t kernel_entry;
|
|
|
|
PCIBus *pci_bus;
|
2011-12-15 22:09:51 +01:00
|
|
|
ISABus *isa_bus;
|
2013-08-03 00:18:51 +02:00
|
|
|
I2CBus *smbus;
|
2012-05-05 13:54:41 +02:00
|
|
|
MIPSCPU *cpu;
|
2012-03-14 01:38:23 +01:00
|
|
|
CPUMIPSState *env;
|
2019-03-09 18:21:40 +01:00
|
|
|
DeviceState *dev;
|
2010-06-29 04:50:43 +02:00
|
|
|
|
|
|
|
/* init CPUs */
|
2017-10-05 15:51:11 +02:00
|
|
|
cpu = MIPS_CPU(cpu_create(machine->cpu_type));
|
2012-05-05 13:54:41 +02:00
|
|
|
env = &cpu->env;
|
2010-06-29 04:50:43 +02:00
|
|
|
|
2012-05-05 13:56:22 +02:00
|
|
|
qemu_register_reset(main_cpu_reset, cpu);
|
2010-06-29 04:50:43 +02:00
|
|
|
|
2019-02-21 13:34:07 +01:00
|
|
|
/* TODO: support more than 256M RAM as highmem */
|
2020-02-19 17:09:25 +01:00
|
|
|
if (machine->ram_size != 256 * MiB) {
|
|
|
|
error_report("Invalid RAM size, should be 256MB");
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
2020-04-26 12:19:16 +02:00
|
|
|
memory_region_add_subregion(address_space_mem, 0, machine->ram);
|
2010-06-29 04:50:43 +02:00
|
|
|
|
2020-04-26 12:19:16 +02:00
|
|
|
/* Boot ROM */
|
2020-04-26 12:16:37 +02:00
|
|
|
memory_region_init_rom(bios, NULL, "fuloong2e.bios", BIOS_SIZE,
|
Fix bad error handling after memory_region_init_ram()
Symptom:
$ qemu-system-x86_64 -m 10000000
Unexpected error in ram_block_add() at /work/armbru/qemu/exec.c:1456:
upstream-qemu: cannot set up guest memory 'pc.ram': Cannot allocate memory
Aborted (core dumped)
Root cause: commit ef701d7 screwed up handling of out-of-memory
conditions. Before the commit, we report the error and exit(1), in
one place, ram_block_add(). The commit lifts the error handling up
the call chain some, to three places. Fine. Except it uses
&error_abort in these places, changing the behavior from exit(1) to
abort(), and thus undoing the work of commit 3922825 "exec: Don't
abort when we can't allocate guest memory".
The three places are:
* memory_region_init_ram()
Commit 4994653 (right after commit ef701d7) lifted the error
handling further, through memory_region_init_ram(), multiplying the
incorrect use of &error_abort. Later on, imitation of existing
(bad) code may have created more.
* memory_region_init_ram_ptr()
The &error_abort is still there.
* memory_region_init_rom_device()
Doesn't need fixing, because commit 33e0eb5 (soon after commit
ef701d7) lifted the error handling further, and in the process
changed it from &error_abort to passing it up the call chain.
Correct, because the callers are realize() methods.
Fix the error handling after memory_region_init_ram() with a
Coccinelle semantic patch:
@r@
expression mr, owner, name, size, err;
position p;
@@
memory_region_init_ram(mr, owner, name, size,
(
- &error_abort
+ &error_fatal
|
err@p
)
);
@script:python@
p << r.p;
@@
print "%s:%s:%s" % (p[0].file, p[0].line, p[0].column)
When the last argument is &error_abort, it gets replaced by
&error_fatal. This is the fix.
If the last argument is anything else, its position is reported. This
lets us check the fix is complete. Four positions get reported:
* ram_backend_memory_alloc()
Error is passed up the call chain, ultimately through
user_creatable_complete(). As far as I can tell, it's callers all
handle the error sanely.
* fsl_imx25_realize(), fsl_imx31_realize(), dp8393x_realize()
DeviceClass.realize() methods, errors handled sanely further up the
call chain.
We're good. Test case again behaves:
$ qemu-system-x86_64 -m 10000000
qemu-system-x86_64: cannot set up guest memory 'pc.ram': Cannot allocate memory
[Exit 1 ]
The next commits will repair the rest of commit ef701d7's damage.
Signed-off-by: Markus Armbruster <armbru@redhat.com>
Message-Id: <1441983105-26376-3-git-send-email-armbru@redhat.com>
Reviewed-by: Peter Crosthwaite <crosthwaite.peter@gmail.com>
2015-09-11 16:51:43 +02:00
|
|
|
&error_fatal);
|
2011-08-08 20:46:13 +02:00
|
|
|
memory_region_add_subregion(address_space_mem, 0x1fc00000LL, bios);
|
2010-06-29 04:50:43 +02:00
|
|
|
|
2019-08-19 14:07:52 +02:00
|
|
|
/*
|
|
|
|
* We do not support flash operation, just loading pmon.bin as raw BIOS.
|
|
|
|
* Please use -L to set the BIOS path and -bios to set bios name.
|
|
|
|
*/
|
2010-06-29 04:50:43 +02:00
|
|
|
|
|
|
|
if (kernel_filename) {
|
2020-02-19 17:09:25 +01:00
|
|
|
loaderparams.ram_size = machine->ram_size;
|
2010-06-29 04:50:43 +02:00
|
|
|
loaderparams.kernel_filename = kernel_filename;
|
|
|
|
loaderparams.kernel_cmdline = kernel_cmdline;
|
|
|
|
loaderparams.initrd_filename = initrd_filename;
|
2019-08-19 14:07:52 +02:00
|
|
|
kernel_entry = load_kernel(env);
|
2011-08-08 20:46:13 +02:00
|
|
|
write_bootloader(env, memory_region_get_ram_ptr(bios), kernel_entry);
|
2010-06-29 04:50:43 +02:00
|
|
|
} else {
|
2010-06-30 19:39:45 +02:00
|
|
|
if (bios_name == NULL) {
|
2020-04-26 12:16:37 +02:00
|
|
|
bios_name = FULOONG_BIOSNAME;
|
2010-06-29 04:50:43 +02:00
|
|
|
}
|
|
|
|
filename = qemu_find_file(QEMU_FILE_TYPE_BIOS, bios_name);
|
|
|
|
if (filename) {
|
|
|
|
bios_size = load_image_targphys(filename, 0x1fc00000LL,
|
|
|
|
BIOS_SIZE);
|
2011-08-21 05:09:37 +02:00
|
|
|
g_free(filename);
|
2010-06-29 04:50:43 +02:00
|
|
|
} else {
|
|
|
|
bios_size = -1;
|
|
|
|
}
|
|
|
|
|
2013-07-29 16:05:30 +02:00
|
|
|
if ((bios_size < 0 || bios_size > BIOS_SIZE) &&
|
|
|
|
!kernel_filename && !qtest_enabled()) {
|
2013-08-03 16:03:18 +02:00
|
|
|
error_report("Could not load MIPS bios '%s'", bios_name);
|
|
|
|
exit(1);
|
2010-06-30 19:39:45 +02:00
|
|
|
}
|
2010-06-29 04:50:43 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Init internal devices */
|
2016-03-15 14:32:19 +01:00
|
|
|
cpu_mips_irq_init_cpu(cpu);
|
|
|
|
cpu_mips_clock_init(cpu);
|
2010-06-29 04:50:43 +02:00
|
|
|
|
|
|
|
/* North bridge, Bonito --> IP2 */
|
|
|
|
pci_bus = bonito_init((qemu_irq *)&(env->irq[2]));
|
|
|
|
|
2018-03-08 23:39:38 +01:00
|
|
|
/* South bridge -> IP5 */
|
2020-04-26 12:16:37 +02:00
|
|
|
vt82c686b_southbridge_init(pci_bus, FULOONG2E_VIA_SLOT, env->irq[5],
|
2018-03-08 23:39:38 +01:00
|
|
|
&smbus, &isa_bus);
|
2010-06-29 04:50:43 +02:00
|
|
|
|
2019-03-09 18:21:40 +01:00
|
|
|
/* GPU */
|
2019-03-19 15:37:19 +01:00
|
|
|
if (vga_interface_type != VGA_NONE) {
|
|
|
|
dev = DEVICE(pci_create(pci_bus, -1, "ati-vga"));
|
|
|
|
qdev_prop_set_uint32(dev, "vgamem_mb", 16);
|
|
|
|
qdev_prop_set_uint16(dev, "x-device-id", 0x5159);
|
|
|
|
qdev_init_nofail(dev);
|
|
|
|
}
|
2019-03-09 18:21:40 +01:00
|
|
|
|
2019-02-21 13:34:07 +01:00
|
|
|
/* Populate SPD eeprom data */
|
smbus: Fix spd_data_generate() error API violation
The Error ** argument must be NULL, &error_abort, &error_fatal, or a
pointer to a variable containing NULL. Passing an argument of the
latter kind twice without clearing it in between is wrong: if the
first call sets an error, it no longer points to NULL for the second
call.
spd_data_generate() can pass @errp to error_setg() more than once when
it adjusts both memory size and type. Harmless, because no caller
passes anything that needs adjusting. Until the previous commit,
sam460ex passed types that needed adjusting, but not sizes.
spd_data_generate()'s contract is rather awkward:
If everything's fine, return non-null and don't set an error.
Else, if memory size or type need adjusting, return non-null and
set an error describing the adjustment.
Else, return null and set an error reporting why no data can be
generated.
Its callers treat the error as a warning even when null is returned.
They don't create the "smbus-eeprom" device then. Suspicious.
Since the previous commit, only "everything's fine" can actually
happen. Drop the unused code and simplify the callers. This gets rid
of the error API violation.
Signed-off-by: Markus Armbruster <armbru@redhat.com>
Message-Id: <20200422134815.1584-3-armbru@redhat.com>
2020-04-22 15:48:13 +02:00
|
|
|
spd_data = spd_data_generate(DDR, machine->ram_size);
|
|
|
|
smbus_eeprom_init_one(smbus, 0x50, spd_data);
|
2010-06-29 04:50:43 +02:00
|
|
|
|
2017-10-17 18:44:16 +02:00
|
|
|
mc146818_rtc_init(isa_bus, 2000, NULL);
|
2010-06-29 04:50:43 +02:00
|
|
|
|
2018-03-08 23:39:38 +01:00
|
|
|
/* Network card: RTL8139D */
|
2013-06-06 10:48:51 +02:00
|
|
|
network_init(pci_bus);
|
2010-06-29 04:50:43 +02:00
|
|
|
}
|
|
|
|
|
2020-04-26 12:16:37 +02:00
|
|
|
static void mips_fuloong2e_machine_init(MachineClass *mc)
|
2010-06-29 04:50:43 +02:00
|
|
|
{
|
2020-04-26 12:16:37 +02:00
|
|
|
mc->desc = "Fuloong 2e mini pc";
|
|
|
|
mc->alias = "fulong2e"; /* Incorrect name used up to QEMU 4.2 */
|
|
|
|
mc->init = mips_fuloong2e_init;
|
2017-02-15 11:05:40 +01:00
|
|
|
mc->block_default_type = IF_IDE;
|
2017-10-05 15:51:11 +02:00
|
|
|
mc->default_cpu_type = MIPS_CPU_TYPE_NAME("Loongson-2E");
|
2019-02-21 13:34:07 +01:00
|
|
|
mc->default_ram_size = 256 * MiB;
|
2020-04-26 12:16:37 +02:00
|
|
|
mc->default_ram_id = "fuloong2e.ram";
|
2020-04-08 11:16:19 +02:00
|
|
|
mc->minimum_page_bits = 14;
|
2010-06-29 04:50:43 +02:00
|
|
|
}
|
|
|
|
|
2020-04-26 12:16:37 +02:00
|
|
|
DEFINE_MACHINE("fuloong2e", mips_fuloong2e_machine_init)
|