2017-06-08 19:09:51 -04:00
|
|
|
//
|
|
|
|
// ZX8081.cpp
|
|
|
|
// Clock Signal
|
|
|
|
//
|
|
|
|
// Created by Thomas Harte on 08/06/2017.
|
2018-05-13 15:19:52 -04:00
|
|
|
// Copyright 2017 Thomas Harte. All rights reserved.
|
2017-06-08 19:09:51 -04:00
|
|
|
//
|
|
|
|
|
|
|
|
#include "ZX8081.hpp"
|
|
|
|
|
2017-06-08 19:49:18 -04:00
|
|
|
using namespace Storage::Data::ZX8081;
|
|
|
|
|
2017-11-11 15:28:40 -05:00
|
|
|
static uint16_t short_at(std::size_t address, const std::vector<uint8_t> &data) {
|
2020-05-09 23:00:39 -04:00
|
|
|
return uint16_t(data[address] | (data[address + 1] << 8));
|
2017-06-08 19:49:18 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
static std::shared_ptr<File> ZX80FileFromData(const std::vector<uint8_t> &data) {
|
|
|
|
// Does this look like a ZX80 file?
|
|
|
|
|
2017-06-08 21:31:03 -04:00
|
|
|
if(data.size() < 0x28) return nullptr;
|
|
|
|
|
2017-06-08 19:49:18 -04:00
|
|
|
// uint16_t next_line_number = short_at(0x2, data);
|
|
|
|
// uint16_t first_visible_line = short_at(0x13, data);
|
|
|
|
|
|
|
|
uint16_t vars = short_at(0x8, data);
|
|
|
|
uint16_t end_of_file = short_at(0xa, data);
|
|
|
|
uint16_t display_address = short_at(0xc, data);
|
|
|
|
|
|
|
|
// check that the end of file is contained within the supplied data
|
2020-05-09 23:00:39 -04:00
|
|
|
if(size_t(end_of_file - 0x4000) > data.size()) return nullptr;
|
2017-06-08 19:49:18 -04:00
|
|
|
|
|
|
|
// check for the proper ordering of buffers
|
|
|
|
if(vars > end_of_file) return nullptr;
|
|
|
|
if(end_of_file > display_address) return nullptr;
|
|
|
|
|
2017-06-08 21:52:13 -04:00
|
|
|
// TODO: does it make sense to inspect the tokenised BASIC?
|
|
|
|
// It starts at 0x4028 and proceeds as [16-bit line number] [tokens] [0x76],
|
|
|
|
// but I'm as yet unable to find documentation of the tokens.
|
|
|
|
|
2017-06-08 19:49:18 -04:00
|
|
|
// TODO: check that the line numbers declared above exist (?)
|
|
|
|
|
2019-12-21 23:52:04 -05:00
|
|
|
auto file = std::make_shared<File>();
|
2017-06-08 19:49:18 -04:00
|
|
|
file->data = data;
|
|
|
|
file->isZX81 = false;
|
|
|
|
return file;
|
|
|
|
}
|
|
|
|
|
2017-06-12 19:41:59 -04:00
|
|
|
static std::shared_ptr<File> ZX81FileFromData(const std::vector<uint8_t> &data) {
|
|
|
|
// Does this look like a ZX81 file?
|
|
|
|
|
|
|
|
// Look for a file name.
|
2017-11-11 15:28:40 -05:00
|
|
|
std::size_t data_pointer = 0;
|
2017-07-17 19:52:54 -04:00
|
|
|
std::vector<uint8_t> name_data;
|
2017-11-11 15:28:40 -05:00
|
|
|
std::size_t c = 11;
|
2017-07-12 21:34:08 -04:00
|
|
|
while(c < data.size() && c--) {
|
2017-07-17 19:52:54 -04:00
|
|
|
name_data.push_back(data[data_pointer] & 0x3f);
|
2017-06-12 19:41:59 -04:00
|
|
|
if(data[data_pointer] & 0x80) break;
|
|
|
|
data_pointer++;
|
|
|
|
}
|
|
|
|
if(!c) return nullptr;
|
|
|
|
data_pointer++;
|
|
|
|
|
|
|
|
if(data.size() < data_pointer + 0x405e - 0x4009) return nullptr;
|
|
|
|
|
2017-07-10 20:44:13 -04:00
|
|
|
// if(data[data_pointer]) return nullptr;
|
2017-06-12 19:41:59 -04:00
|
|
|
|
2018-05-27 13:31:30 -04:00
|
|
|
// uint16_t vars = short_at(data_pointer + 0x4010 - 0x4009, data);
|
2017-06-12 19:41:59 -04:00
|
|
|
uint16_t end_of_file = short_at(data_pointer + 0x4014 - 0x4009, data);
|
|
|
|
// uint16_t display_address = short_at(0x400c - 0x4009, data);
|
|
|
|
|
|
|
|
// check that the end of file is contained within the supplied data
|
|
|
|
if(data_pointer + end_of_file - 0x4009 > data.size()) return nullptr;
|
|
|
|
|
|
|
|
// check for the proper ordering of buffers
|
2018-05-26 19:05:35 -04:00
|
|
|
// if(vars > end_of_file) return nullptr;
|
2017-06-12 19:41:59 -04:00
|
|
|
// if(end_of_file > display_address) return nullptr;
|
|
|
|
|
|
|
|
// TODO: does it make sense to inspect the tokenised BASIC?
|
|
|
|
// It starts at 0x4028 and proceeds as [16-bit line number] [tokens] [0x76],
|
|
|
|
// but I'm as yet unable to find documentation of the tokens.
|
|
|
|
|
|
|
|
// TODO: check that the line numbers declared above exist (?)
|
|
|
|
|
2019-12-21 23:52:04 -05:00
|
|
|
auto file = std::make_shared<File>();
|
2017-07-17 19:52:54 -04:00
|
|
|
file->name = StringFromData(name_data, true);
|
2017-06-12 19:41:59 -04:00
|
|
|
file->data = data;
|
|
|
|
file->isZX81 = true;
|
|
|
|
return file;
|
|
|
|
}
|
|
|
|
|
2017-06-08 19:49:18 -04:00
|
|
|
std::shared_ptr<File> Storage::Data::ZX8081::FileFromData(const std::vector<uint8_t> &data) {
|
2017-06-22 20:23:14 -04:00
|
|
|
std::shared_ptr<Storage::Data::ZX8081::File> result = ZX81FileFromData(data);
|
2017-06-12 19:41:59 -04:00
|
|
|
if(result) return result;
|
2017-06-22 20:23:14 -04:00
|
|
|
return ZX80FileFromData(data);
|
2017-06-08 19:09:51 -04:00
|
|
|
}
|
2017-06-12 21:32:36 -04:00
|
|
|
|
2017-11-12 15:59:11 -05:00
|
|
|
// MARK: - String conversion
|
2017-06-12 21:32:36 -04:00
|
|
|
|
|
|
|
std::wstring Storage::Data::ZX8081::StringFromData(const std::vector<uint8_t> &data, bool is_zx81) {
|
|
|
|
std::wstring string;
|
|
|
|
|
2017-06-12 22:08:11 -04:00
|
|
|
wchar_t zx80_map[64] = {
|
2018-05-13 15:50:56 -04:00
|
|
|
' ', u'\u2598', u'\u259d', u'\u2580', u'\u2596', u'\u258c', u'\u259e', u'\u259b', u'\u2588', u'\u2584', u'\u2580', '"', u'\u00a3', '$', ':', '?',
|
|
|
|
'(', ')', '>', '<', '=', '+', '-', '*', '/', ';', ',', '.', '0', '1', '2', '3',
|
|
|
|
'4', '5', '6', '7', '8', '9', 'A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'I', 'J',
|
|
|
|
'K', 'L', 'M', 'N', 'O', 'P', 'Q', 'R', 'S', 'T', 'U', 'V', 'W', 'X', 'Y', 'Z'
|
2017-06-12 21:32:36 -04:00
|
|
|
};
|
2017-06-12 22:08:11 -04:00
|
|
|
// TODO: the block character conversions shown here are in the wrong order
|
|
|
|
wchar_t zx81_map[64] = {
|
2018-05-13 15:50:56 -04:00
|
|
|
' ', u'\u2598', u'\u259d', u'\u2580', u'\u2596', u'\u258c', u'\u259e', u'\u259b', u'\u2588', u'\u2584', u'\u2580', '"', u'\u00a3', '$', ':', '?',
|
|
|
|
'(', ')', '-', '+', '*', '/', '=', '>', '<', ';', ',', '.', '0', '1', '2', '3',
|
|
|
|
'4', '5', '6', '7', '8', '9', 'A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'I', 'J',
|
|
|
|
'K', 'L', 'M', 'N', 'O', 'P', 'Q', 'R', 'S', 'T', 'U', 'V', 'W', 'X', 'Y', 'Z'
|
2017-06-12 21:32:36 -04:00
|
|
|
};
|
|
|
|
wchar_t *map = is_zx81 ? zx81_map : zx80_map;
|
|
|
|
|
|
|
|
for(uint8_t byte : data) {
|
|
|
|
string.push_back(map[byte & 0x3f]);
|
|
|
|
}
|
|
|
|
|
|
|
|
return string;
|
|
|
|
}
|
|
|
|
|
2017-07-21 21:24:28 -04:00
|
|
|
std::vector<uint8_t> Storage::Data::ZX8081::DataFromString(const std::wstring &string, bool is_zx81) {
|
2017-06-12 21:32:36 -04:00
|
|
|
std::vector<uint8_t> data;
|
|
|
|
|
|
|
|
// TODO
|
2020-05-30 01:06:43 -04:00
|
|
|
(void)string;
|
|
|
|
(void)is_zx81;
|
2017-06-12 21:32:36 -04:00
|
|
|
|
|
|
|
return data;
|
|
|
|
}
|