]>
Commit | Line | Data |
---|---|---|
c0abd68b | 1 | /* |
5e8ac5e0 | 2 | * (C) Copyright 2015,2016,2018 Leo C. <erbl259-lmu@yahoo.de> |
c0abd68b | 3 | * |
b51d2360 L |
4 | * SPDX-License-Identifier: GPL-2.0 |
5 | */ | |
6 | ||
7 | /* | |
8 | * See CP/M 3 System Manual, Appendix D: CPM3.SYS File Format | |
c0abd68b L |
9 | */ |
10 | ||
7eecbdac | 11 | #include "cmd_loadcpm3.h" |
c0abd68b | 12 | #include <ctype.h> |
c0abd68b | 13 | |
c0abd68b L |
14 | #include "env.h" |
15 | #include "ff.h" | |
2d914b45 | 16 | #include "eval_arg.h" |
c0abd68b L |
17 | #include "con-utils.h" |
18 | #include "z80-if.h" | |
19 | #include "debug.h" | |
5e8ac5e0 | 20 | #include "errnum.h" |
c0abd68b L |
21 | |
22 | ||
23 | #define RS 128 /* CP/M record size */ | |
24 | ||
b51d2360 L |
25 | #define FSIZE_t DWORD |
26 | ||
5e8ac5e0 L |
27 | static FRESULT read_record(FIL *fp, uint8_t *buffer) |
28 | { | |
29 | unsigned int br; /* bytes read */ | |
30 | ||
31 | FRESULT res = f_read(fp, buffer, RS, &br); | |
32 | if (br != RS) | |
33 | return EEOF; | |
34 | return res; | |
35 | } | |
36 | ||
c0abd68b L |
37 | /* |
38 | * Load Routine | |
39 | * | |
40 | * Input: addr = Page Address of load top | |
41 | * len = Length in pages of module to read | |
42 | * | |
43 | */ | |
5e8ac5e0 | 44 | ERRNUM load(FIL *File, uint32_t addr, uint8_t len) |
c0abd68b L |
45 | { |
46 | uint8_t buffer[RS]; | |
c0abd68b L |
47 | |
48 | len *= 2; /* length in records of module */ | |
49 | //debug("## load: addr: 0x%.4X, records: 0x%.4X, (%u)\n", addr, len, len); | |
50 | ||
51 | for (; len; len--) { | |
52 | addr -= RS; | |
5e8ac5e0 L |
53 | FRESULT res = read_record(File, buffer); |
54 | if (res) | |
55 | return res; | |
c0abd68b | 56 | |
5e8ac5e0 L |
57 | if (!(z80_bus_cmd(Request) & ZST_ACQUIRED)) |
58 | return EBUSTO; | |
c0abd68b L |
59 | z80_write_block(buffer, addr, RS); |
60 | z80_bus_cmd(Release); | |
61 | //debug("## written: 0x%.4X\n", addr); | |
62 | } | |
63 | ||
5e8ac5e0 | 64 | return ESUCCESS; |
c0abd68b L |
65 | } |
66 | ||
ad9bc17c | 67 | #pragma GCC diagnostic ignored "-Wmaybe-uninitialized" |
c0abd68b | 68 | |
5e8ac5e0 | 69 | command_ret_t do_loadcpm3(cmd_tbl_t *cmdtp UNUSED, uint_fast8_t flag UNUSED, int argc, char * const argv[]) |
c0abd68b L |
70 | { |
71 | uint16_t mem_top; | |
72 | uint8_t res_len; | |
73 | uint16_t bank_top; | |
74 | uint8_t bank_len; | |
057817cb | 75 | uint16_t osentry_addr = 0; |
b51d2360 L |
76 | uint32_t common_base = 0; |
77 | uint32_t banked_base; | |
c0abd68b | 78 | char *fname; |
c0abd68b | 79 | FIL File; |
209025e8 | 80 | char default_fname[strlen_P(PSTR(CONFIG_CPM3_SYSFILE)) + 1]; |
c0abd68b | 81 | uint8_t buffer[RS]; |
5e8ac5e0 | 82 | FRESULT res; |
c0abd68b L |
83 | |
84 | ||
b51d2360 L |
85 | //common_base = getenv_ulong(PSTR(ENV_CPM3_COMMON_BASE), 16, |
86 | // CONFIG_CPM3_COMMON_BASE); | |
057817cb L |
87 | banked_base = getenv_ulong(PSTR(ENV_CPM3_BANKED_BASE), 16, |
88 | CONFIG_CPM3_BANKED_BASE); | |
c0abd68b | 89 | |
057817cb | 90 | if (argc > 3) |
2d914b45 | 91 | banked_base = eval_arg(argv[3], NULL); |
057817cb | 92 | if (argc > 2) |
2d914b45 | 93 | common_base = eval_arg(argv[2], NULL); |
c0abd68b | 94 | |
bddc7e77 | 95 | fname = getenv_str(PSTR(ENV_CPM3_SYSFILE)); |
209025e8 L |
96 | if (fname == NULL || *fname == '\0') { |
97 | strcpy_P(default_fname, PSTR(CONFIG_CPM3_SYSFILE)); | |
98 | fname = default_fname; | |
99 | } | |
98aac4d0 L |
100 | if (argc > 1) { |
101 | fname = argv[1]; | |
c0abd68b | 102 | } |
c0abd68b | 103 | |
bbd45c46 | 104 | res = f_open(&File, fname, FA_READ ); |
5e8ac5e0 L |
105 | if (res) |
106 | goto out; | |
c0abd68b L |
107 | |
108 | printf_P(PSTR("Loading: '%s'...\n"), fname); | |
109 | ||
110 | /* read the load record */ | |
5e8ac5e0 L |
111 | res = read_record(&File, buffer); |
112 | if (res) | |
c0abd68b L |
113 | goto out; |
114 | ||
115 | mem_top = buffer[0] << 8; | |
116 | res_len = buffer[1]; | |
117 | bank_top = buffer[2] << 8; | |
118 | bank_len = buffer[3]; | |
057817cb | 119 | osentry_addr = buffer[4] + (buffer[5] << 8); |
c0abd68b L |
120 | |
121 | /* read display info */ | |
5e8ac5e0 L |
122 | res = read_record(&File, buffer); |
123 | if (res) | |
c0abd68b L |
124 | goto out; |
125 | ||
126 | /* print the info */ | |
127 | buffer[RS-1] = '$'; | |
128 | uint8_t *p = memchr(buffer, '$', RS); | |
129 | *p = '\0'; | |
130 | my_puts((char *)buffer); | |
131 | ||
b51d2360 L |
132 | if (common_base == 0) { |
133 | /* read common base | |
134 | * http://www.seasip.info/Cpm/scb.html | |
135 | */ | |
136 | FSIZE_t common_base_ofs = ((res_len - 6) << 8) + 2*RS + RS-7; | |
137 | FSIZE_t cur_pos = f_tell(&File); | |
5e8ac5e0 | 138 | unsigned int br; /* bytes read */ |
b51d2360 L |
139 | if ((res = f_lseek(&File, common_base_ofs)) || |
140 | (res = f_read(&File, buffer, 2, &br)) || | |
b51d2360 L |
141 | (res = f_lseek(&File, cur_pos))) |
142 | goto out; | |
5e8ac5e0 L |
143 | if (br != 2) { |
144 | res = EEOF; | |
145 | goto out; | |
146 | } | |
b51d2360 L |
147 | common_base = (uint16_t) buffer[0] + (buffer[1] << 8); |
148 | setenv_hex(PSTR(ENV_CPM3_COMMON_BASE), common_base); | |
149 | } | |
2b3aff48 L |
150 | |
151 | setenv_hex(PSTR(ENV_CPM3_SCB), mem_top - ((res_len - (6 - 1)) << 8) + common_base); | |
152 | ||
c0abd68b L |
153 | /* Main System Load */ |
154 | ||
155 | /* Load Common Portion of System */ | |
057817cb | 156 | if ((res = load(&File, common_base + mem_top, res_len)) != 0) |
c0abd68b L |
157 | goto out; |
158 | ||
159 | /* Load Banked Portion of System */ | |
057817cb | 160 | res = load(&File, banked_base + bank_top, bank_len); |
c0abd68b L |
161 | |
162 | out: | |
163 | f_close(&File); | |
c0abd68b | 164 | |
5e8ac5e0 L |
165 | if (res) |
166 | cmd_error(CMD_RET_FAILURE, res, PSTR("%s"), fname); | |
167 | ||
168 | if (res_len != 0) { | |
169 | if (osentry_addr + common_base > 0xffff) { | |
170 | z80_bus_cmd(Request); | |
171 | if (z80_read(osentry_addr + common_base) == 0xc3) { | |
172 | osentry_addr = z80_read(osentry_addr+common_base+1) + | |
173 | (z80_read(osentry_addr + common_base+2) << 8); | |
057817cb | 174 | } |
5e8ac5e0 L |
175 | z80_bus_cmd(Release); |
176 | if (banked_base + osentry_addr > 0xffff) | |
177 | osentry_addr = 0; | |
057817cb | 178 | } |
5e8ac5e0 | 179 | setenv_hex(PSTR(ENV_STARTADDRESS), osentry_addr); |
c0abd68b | 180 | } |
5e8ac5e0 L |
181 | printf_P(PSTR("Loaded: Resident: ")); |
182 | if (res_len != 0) | |
183 | printf_P(PSTR("%.5lX-%.5lX, "), | |
184 | (common_base + mem_top) - res_len*256, | |
185 | (common_base + mem_top) - 1); | |
186 | else | |
187 | printf_P(PSTR(" - ")); | |
188 | printf_P(PSTR("Banked: ")); | |
189 | if (bank_len != 0) | |
190 | printf_P(PSTR("%.5lX-%.5lX\n"), | |
191 | (banked_base + bank_top) - bank_len*256, | |
192 | (banked_base + bank_top) - 1); | |
193 | else | |
194 | printf_P(PSTR(" - \n")); | |
195 | ||
196 | return CMD_RET_SUCCESS; | |
c0abd68b | 197 | } |