2016-08-29 12:48:49 +00:00
|
|
|
//
|
|
|
|
// AcornAnalyser.cpp
|
|
|
|
// Clock Signal
|
|
|
|
//
|
|
|
|
// Created by Thomas Harte on 29/08/2016.
|
2018-05-13 19:19:52 +00:00
|
|
|
// Copyright 2016 Thomas Harte. All rights reserved.
|
2016-08-29 12:48:49 +00:00
|
|
|
//
|
|
|
|
|
2016-09-15 23:24:59 +00:00
|
|
|
#include "StaticAnalyser.hpp"
|
2016-08-29 12:48:49 +00:00
|
|
|
|
2016-09-19 01:18:11 +00:00
|
|
|
#include "Disk.hpp"
|
2016-08-30 01:53:06 +00:00
|
|
|
#include "Tape.hpp"
|
2018-03-09 21:07:29 +00:00
|
|
|
#include "Target.hpp"
|
2016-08-30 01:53:06 +00:00
|
|
|
|
2018-01-25 02:48:44 +00:00
|
|
|
using namespace Analyser::Static::Acorn;
|
2016-08-29 12:48:49 +00:00
|
|
|
|
2018-01-24 03:18:16 +00:00
|
|
|
static std::vector<std::shared_ptr<Storage::Cartridge::Cartridge>>
|
|
|
|
AcornCartridgesFrom(const std::vector<std::shared_ptr<Storage::Cartridge::Cartridge>> &cartridges) {
|
|
|
|
std::vector<std::shared_ptr<Storage::Cartridge::Cartridge>> acorn_cartridges;
|
2016-08-29 12:48:49 +00:00
|
|
|
|
2017-12-02 21:01:30 +00:00
|
|
|
for(const auto &cartridge : cartridges) {
|
|
|
|
const auto &segments = cartridge->get_segments();
|
2016-08-29 12:48:49 +00:00
|
|
|
|
|
|
|
// only one mapped item is allowed
|
|
|
|
if(segments.size() != 1) continue;
|
|
|
|
|
2018-01-16 02:27:45 +00:00
|
|
|
// which must be 8 or 16 kb in size
|
2018-01-17 03:27:41 +00:00
|
|
|
const Storage::Cartridge::Cartridge::Segment &segment = segments.front();
|
2018-01-16 02:27:45 +00:00
|
|
|
if(segment.data.size() != 0x4000 && segment.data.size() != 0x2000) continue;
|
2016-08-29 12:48:49 +00:00
|
|
|
|
|
|
|
// is a copyright string present?
|
|
|
|
uint8_t copyright_offset = segment.data[7];
|
|
|
|
if(
|
|
|
|
segment.data[copyright_offset] != 0x00 ||
|
|
|
|
segment.data[copyright_offset+1] != 0x28 ||
|
|
|
|
segment.data[copyright_offset+2] != 0x43 ||
|
|
|
|
segment.data[copyright_offset+3] != 0x29
|
|
|
|
) continue;
|
|
|
|
|
|
|
|
// is the language entry point valid?
|
|
|
|
if(!(
|
|
|
|
(segment.data[0] == 0x00 && segment.data[1] == 0x00 && segment.data[2] == 0x00) ||
|
|
|
|
(segment.data[0] != 0x00 && segment.data[2] >= 0x80 && segment.data[2] < 0xc0)
|
|
|
|
)) continue;
|
|
|
|
|
|
|
|
// is the service entry point valid?
|
|
|
|
if(!(segment.data[5] >= 0x80 && segment.data[5] < 0xc0)) continue;
|
|
|
|
|
|
|
|
// probability of a random binary blob that isn't an Acorn ROM proceeding to here:
|
|
|
|
// 1/(2^32) *
|
|
|
|
// ( ((2^24)-1)/(2^24)*(1/4) + 1/(2^24) ) *
|
|
|
|
// 1/4
|
2018-05-13 19:34:31 +00:00
|
|
|
// = something very improbable, around 1/16th of 1 in 2^32, but not exactly.
|
2016-08-29 12:48:49 +00:00
|
|
|
acorn_cartridges.push_back(cartridge);
|
|
|
|
}
|
|
|
|
|
|
|
|
return acorn_cartridges;
|
|
|
|
}
|
|
|
|
|
2018-04-14 16:12:12 +00:00
|
|
|
Analyser::Static::TargetList Analyser::Static::Acorn::GetTargets(const Media &media, const std::string &file_name, TargetPlatform::IntType potential_platforms) {
|
2018-01-25 03:35:54 +00:00
|
|
|
std::unique_ptr<Target> target(new Target);
|
|
|
|
target->machine = Machine::Electron;
|
2018-03-07 19:24:52 +00:00
|
|
|
target->confidence = 0.5; // TODO: a proper estimation
|
2018-03-09 21:07:29 +00:00
|
|
|
target->has_dfs = false;
|
|
|
|
target->has_adfs = false;
|
|
|
|
target->should_shift_restart = false;
|
2016-09-01 00:24:13 +00:00
|
|
|
|
2016-08-29 12:48:49 +00:00
|
|
|
// strip out inappropriate cartridges
|
2018-01-25 03:35:54 +00:00
|
|
|
target->media.cartridges = AcornCartridgesFrom(media.cartridges);
|
2016-08-29 12:48:49 +00:00
|
|
|
|
2016-08-30 01:10:38 +00:00
|
|
|
// if there are any tapes, attempt to get data from the first
|
2018-07-11 00:00:46 +00:00
|
|
|
if(!media.tapes.empty()) {
|
2017-08-17 14:48:29 +00:00
|
|
|
std::shared_ptr<Storage::Tape::Tape> tape = media.tapes.front();
|
2018-01-24 03:18:16 +00:00
|
|
|
std::vector<File> files = GetFiles(tape);
|
2017-07-13 01:34:08 +00:00
|
|
|
tape->reset();
|
2016-09-01 00:24:13 +00:00
|
|
|
|
|
|
|
// continue if there are any files
|
2018-07-11 00:00:46 +00:00
|
|
|
if(!files.empty()) {
|
2016-09-01 00:24:13 +00:00
|
|
|
bool is_basic = true;
|
|
|
|
|
|
|
|
// protected files are always for *RUNning only
|
2016-09-05 21:55:40 +00:00
|
|
|
if(files.front().is_protected) is_basic = false;
|
2016-09-01 00:24:13 +00:00
|
|
|
|
|
|
|
// check also for a continuous threading of BASIC lines; if none then this probably isn't BASIC code,
|
|
|
|
// so that's also justification to *RUN
|
2017-11-11 20:28:40 +00:00
|
|
|
std::size_t pointer = 0;
|
2016-09-01 00:24:13 +00:00
|
|
|
uint8_t *data = &files.front().data[0];
|
2017-11-11 20:28:40 +00:00
|
|
|
std::size_t data_size = files.front().data.size();
|
2017-03-26 18:34:47 +00:00
|
|
|
while(1) {
|
|
|
|
if(pointer >= data_size-1 || data[pointer] != 13) {
|
2016-09-01 00:24:13 +00:00
|
|
|
is_basic = false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if((data[pointer+1]&0x7f) == 0x7f) break;
|
|
|
|
pointer += data[pointer+3];
|
|
|
|
}
|
|
|
|
|
|
|
|
// Inspect first file. If it's protected or doesn't look like BASIC
|
|
|
|
// then the loading command is *RUN. Otherwise it's CHAIN"".
|
2018-01-25 03:35:54 +00:00
|
|
|
target->loading_command = is_basic ? "CHAIN\"\"\n" : "*RUN\n";
|
2016-09-01 00:24:13 +00:00
|
|
|
|
2018-01-25 03:35:54 +00:00
|
|
|
target->media.tapes = media.tapes;
|
2016-09-01 00:24:13 +00:00
|
|
|
}
|
2016-08-30 01:10:38 +00:00
|
|
|
}
|
2016-09-01 00:24:13 +00:00
|
|
|
|
2018-07-11 00:00:46 +00:00
|
|
|
if(!media.disks.empty()) {
|
2017-08-17 14:48:29 +00:00
|
|
|
std::shared_ptr<Storage::Disk::Disk> disk = media.disks.front();
|
2016-09-25 21:46:11 +00:00
|
|
|
std::unique_ptr<Catalogue> dfs_catalogue, adfs_catalogue;
|
|
|
|
dfs_catalogue = GetDFSCatalogue(disk);
|
|
|
|
if(dfs_catalogue == nullptr) adfs_catalogue = GetADFSCatalogue(disk);
|
2017-03-26 18:34:47 +00:00
|
|
|
if(dfs_catalogue || adfs_catalogue) {
|
2018-01-25 03:35:54 +00:00
|
|
|
target->media.disks = media.disks;
|
2018-03-09 21:07:29 +00:00
|
|
|
target->has_dfs = !!dfs_catalogue;
|
|
|
|
target->has_adfs = !!adfs_catalogue;
|
2016-09-19 12:29:23 +00:00
|
|
|
|
2016-09-29 01:28:34 +00:00
|
|
|
Catalogue::BootOption bootOption = (dfs_catalogue ?: adfs_catalogue)->bootOption;
|
2017-01-08 19:46:19 +00:00
|
|
|
if(bootOption != Catalogue::BootOption::None)
|
2018-03-09 21:07:29 +00:00
|
|
|
target->should_shift_restart = true;
|
2017-01-08 19:46:19 +00:00
|
|
|
else
|
2018-01-25 03:35:54 +00:00
|
|
|
target->loading_command = "*CAT\n";
|
2016-09-19 12:29:23 +00:00
|
|
|
}
|
2016-09-19 01:18:11 +00:00
|
|
|
}
|
2016-09-01 00:24:13 +00:00
|
|
|
|
2018-04-14 16:12:12 +00:00
|
|
|
TargetList targets;
|
|
|
|
if(!target->media.empty()) {
|
|
|
|
targets.push_back(std::move(target));
|
2018-03-09 21:07:29 +00:00
|
|
|
}
|
2018-04-14 16:12:12 +00:00
|
|
|
return targets;
|
2016-08-30 01:10:38 +00:00
|
|
|
}
|