/* * VMApple Configuration Region * * Copyright © 2023 Amazon.com, Inc. or its affiliates. All Rights Reserved. * * SPDX-License-Identifier: GPL-2.0-or-later * * This work is licensed under the terms of the GNU GPL, version 2 or later. * See the COPYING file in the top-level directory. */ #include "qemu/osdep.h" #include "hw/vmapple/vmapple.h" #include "hw/sysbus.h" #include "qemu/log.h" #include "qemu/module.h" #include "qapi/error.h" #include "net/net.h" OBJECT_DECLARE_SIMPLE_TYPE(VMAppleCfgState, VMAPPLE_CFG) #define VMAPPLE_CFG_SIZE 0x00010000 typedef struct VMAppleCfg { uint32_t version; /* 0x000 */ uint32_t nr_cpus; /* 0x004 */ uint32_t unk1; /* 0x008 */ uint32_t unk2; /* 0x00c */ uint32_t unk3; /* 0x010 */ uint32_t unk4; /* 0x014 */ uint64_t ecid; /* 0x018 */ uint64_t ram_size; /* 0x020 */ uint32_t run_installer1; /* 0x028 */ uint32_t unk5; /* 0x02c */ uint32_t unk6; /* 0x030 */ uint32_t run_installer2; /* 0x034 */ uint32_t rnd; /* 0x038 */ uint32_t unk7; /* 0x03c */ MACAddr mac_en0; /* 0x040 */ uint8_t pad1[2]; MACAddr mac_en1; /* 0x048 */ uint8_t pad2[2]; MACAddr mac_wifi0; /* 0x050 */ uint8_t pad3[2]; MACAddr mac_bt0; /* 0x058 */ uint8_t pad4[2]; uint8_t reserved[0xa0]; /* 0x060 */ uint32_t cpu_ids[0x80]; /* 0x100 */ uint8_t scratch[0x200]; /* 0x180 */ char serial[32]; /* 0x380 */ char unk8[32]; /* 0x3a0 */ char model[32]; /* 0x3c0 */ uint8_t unk9[32]; /* 0x3e0 */ uint32_t unk10; /* 0x400 */ char soc_name[32]; /* 0x404 */ } VMAppleCfg; struct VMAppleCfgState { SysBusDevice parent_obj; VMAppleCfg cfg; MemoryRegion mem; char *serial; char *model; char *soc_name; }; static void vmapple_cfg_reset(Object *obj, ResetType type) { VMAppleCfgState *s = VMAPPLE_CFG(obj); VMAppleCfg *cfg; cfg = memory_region_get_ram_ptr(&s->mem); memset(cfg, 0, VMAPPLE_CFG_SIZE); *cfg = s->cfg; } static bool set_fixlen_property_or_error(char *restrict dst, const char *restrict src, size_t dst_size, Error **errp, const char *property_name) { ERRP_GUARD(); size_t len; len = g_strlcpy(dst, src, dst_size); if (len < dst_size) { /* len does not count nul terminator */ return true; } error_setg(errp, "Provided value too long for property '%s'", property_name); error_append_hint(errp, "length (%zu) exceeds maximum of %zu\n", len, dst_size - 1); return false; } #define set_fixlen_property_or_return(dst_array, src, errp, property_name) \ do { \ if (!set_fixlen_property_or_error((dst_array), (src), \ ARRAY_SIZE(dst_array), \ (errp), (property_name))) { \ return; \ } \ } while (0) static void vmapple_cfg_realize(DeviceState *dev, Error **errp) { VMAppleCfgState *s = VMAPPLE_CFG(dev); uint32_t i; if (!s->serial) { s->serial = g_strdup("1234"); } if (!s->model) { s->model = g_strdup("VM0001"); } if (!s->soc_name) { s->soc_name = g_strdup("Apple M1 (Virtual)"); } set_fixlen_property_or_return(s->cfg.serial, s->serial, errp, "serial"); set_fixlen_property_or_return(s->cfg.model, s->model, errp, "model"); set_fixlen_property_or_return(s->cfg.soc_name, s->soc_name, errp, "soc_name"); set_fixlen_property_or_return(s->cfg.unk8, "D/A", errp, "unk8"); s->cfg.version = 2; s->cfg.unk1 = 1; s->cfg.unk2 = 1; s->cfg.unk3 = 0x20; s->cfg.unk4 = 0; s->cfg.unk5 = 1; s->cfg.unk6 = 1; s->cfg.unk7 = 0; s->cfg.unk10 = 1; if (s->cfg.nr_cpus > ARRAY_SIZE(s->cfg.cpu_ids)) { error_setg(errp, "Failed to create %u CPUs, vmapple machine supports %zu max", s->cfg.nr_cpus, ARRAY_SIZE(s->cfg.cpu_ids)); return; } for (i = 0; i < s->cfg.nr_cpus; i++) { s->cfg.cpu_ids[i] = i; } } static void vmapple_cfg_init(Object *obj) { VMAppleCfgState *s = VMAPPLE_CFG(obj); memory_region_init_ram(&s->mem, obj, "VMApple Config", VMAPPLE_CFG_SIZE, &error_fatal); sysbus_init_mmio(SYS_BUS_DEVICE(obj), &s->mem); } static const Property vmapple_cfg_properties[] = { DEFINE_PROP_UINT32("nr-cpus", VMAppleCfgState, cfg.nr_cpus, 1), DEFINE_PROP_UINT64("ecid", VMAppleCfgState, cfg.ecid, 0), DEFINE_PROP_UINT64("ram-size", VMAppleCfgState, cfg.ram_size, 0), DEFINE_PROP_UINT32("run_installer1", VMAppleCfgState, cfg.run_installer1, 0), DEFINE_PROP_UINT32("run_installer2", VMAppleCfgState, cfg.run_installer2, 0), DEFINE_PROP_UINT32("rnd", VMAppleCfgState, cfg.rnd, 0), DEFINE_PROP_MACADDR("mac-en0", VMAppleCfgState, cfg.mac_en0), DEFINE_PROP_MACADDR("mac-en1", VMAppleCfgState, cfg.mac_en1), DEFINE_PROP_MACADDR("mac-wifi0", VMAppleCfgState, cfg.mac_wifi0), DEFINE_PROP_MACADDR("mac-bt0", VMAppleCfgState, cfg.mac_bt0), DEFINE_PROP_STRING("serial", VMAppleCfgState, serial), DEFINE_PROP_STRING("model", VMAppleCfgState, model), DEFINE_PROP_STRING("soc_name", VMAppleCfgState, soc_name), }; static void vmapple_cfg_class_init(ObjectClass *klass, void *data) { DeviceClass *dc = DEVICE_CLASS(klass); ResettableClass *rc = RESETTABLE_CLASS(klass); dc->realize = vmapple_cfg_realize; dc->desc = "VMApple Configuration Region"; device_class_set_props(dc, vmapple_cfg_properties); rc->phases.hold = vmapple_cfg_reset; } static const TypeInfo vmapple_cfg_info = { .name = TYPE_VMAPPLE_CFG, .parent = TYPE_SYS_BUS_DEVICE, .instance_size = sizeof(VMAppleCfgState), .instance_init = vmapple_cfg_init, .class_init = vmapple_cfg_class_init, }; static void vmapple_cfg_register_types(void) { type_register_static(&vmapple_cfg_info); } type_init(vmapple_cfg_register_types)