2019-12-10 14:01:22 +11:00
|
|
|
#include <stdlib.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <fcntl.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
|
|
|
|
#include "common.h"
|
|
|
|
|
2020-04-14 00:25:27 +10:00
|
|
|
/* Push specified file to specified device running Forth and verify
|
2019-12-10 14:01:22 +11:00
|
|
|
* that the sent contents is correct.
|
|
|
|
*/
|
|
|
|
|
|
|
|
int main(int argc, char **argv)
|
|
|
|
{
|
|
|
|
if (argc != 4) {
|
|
|
|
fprintf(stderr, "Usage: ./upload device memptr fname\n");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
unsigned int memptr = strtol(argv[2], NULL, 16);
|
|
|
|
FILE *fp = fopen(argv[3], "r");
|
|
|
|
if (!fp) {
|
|
|
|
fprintf(stderr, "Can't open %s.\n", argv[3]);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
fseek(fp, 0, SEEK_END);
|
|
|
|
unsigned int bytecount = ftell(fp);
|
|
|
|
fprintf(stderr, "memptr: 0x%04x bytecount: 0x%04x.\n", memptr, bytecount);
|
2019-12-12 01:24:40 +11:00
|
|
|
if (!bytecount) {
|
|
|
|
// Nothing to read
|
|
|
|
fclose(fp);
|
|
|
|
return 0;
|
|
|
|
}
|
2019-12-10 14:01:22 +11:00
|
|
|
if (memptr+bytecount > 0xffff) {
|
|
|
|
fprintf(stderr, "memptr+bytecount out of range.\n");
|
|
|
|
fclose(fp);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
rewind(fp);
|
2020-07-03 01:36:53 +10:00
|
|
|
int fd = ttyopen(argv[1]);
|
2020-04-14 00:25:27 +10:00
|
|
|
if (fd < 0) {
|
|
|
|
fprintf(stderr, "Could not open %s\n", argv[1]);
|
|
|
|
return 1;
|
|
|
|
}
|
2019-12-13 04:04:56 +11:00
|
|
|
char s[0x40];
|
2020-04-14 00:25:27 +10:00
|
|
|
sprintf(s,
|
Rename A!, A@ and A, to C!*, C@* and C,*
Also, rename "Addressed devices" to "Indirect memory access".
I do this because I need to add indirect versions of !, @ and ,
to allow boostrapping directly to EEPROM and that A,, thing I've
added to assemblers felt like really bad names.
With this change, I'd like to generalize the use of the * suffix
for aliases.
2020-12-08 11:06:58 +11:00
|
|
|
": _ 0x%04x 0x%04x DO KEY DUP .x I C!* LOOP ; _",
|
2020-04-14 00:25:27 +10:00
|
|
|
memptr+bytecount, memptr);
|
2019-12-13 04:04:56 +11:00
|
|
|
sendcmd(fd, s);
|
2019-12-12 01:24:40 +11:00
|
|
|
|
|
|
|
int returncode = 0;
|
2019-12-10 14:01:22 +11:00
|
|
|
while (fread(s, 1, 1, fp)) {
|
2020-07-03 01:36:53 +10:00
|
|
|
putc('.', stderr);
|
|
|
|
fflush(stderr);
|
2019-12-10 14:01:22 +11:00
|
|
|
unsigned char c = s[0];
|
|
|
|
write(fd, &c, 1);
|
2019-12-12 01:24:40 +11:00
|
|
|
usleep(1000); // let it breathe
|
2020-04-14 00:25:27 +10:00
|
|
|
mread(fd, s, 2); // read hex pair
|
2019-12-10 14:01:22 +11:00
|
|
|
s[2] = 0; // null terminate
|
|
|
|
unsigned char c2 = strtol(s, NULL, 16);
|
|
|
|
if (c != c2) {
|
|
|
|
// mismatch!
|
|
|
|
unsigned int pos = ftell(fp);
|
|
|
|
fprintf(stderr, "Mismatch at byte %d! %d != %d.\n", pos, c, c2);
|
2019-12-12 01:24:40 +11:00
|
|
|
// we don't exit now because we need to "consume" our whole program.
|
|
|
|
returncode = 1;
|
2019-12-10 14:01:22 +11:00
|
|
|
}
|
2020-04-14 04:41:02 +10:00
|
|
|
usleep(1000); // let it breathe
|
2019-12-10 14:01:22 +11:00
|
|
|
}
|
2020-05-02 04:39:13 +10:00
|
|
|
readprompt(fd);
|
2020-04-14 00:25:27 +10:00
|
|
|
sendcmdp(fd, "FORGET _");
|
2020-07-03 01:36:53 +10:00
|
|
|
fprintf(stderr, "Done!\n");
|
2019-12-12 01:24:40 +11:00
|
|
|
fclose(fp);
|
2020-07-03 01:36:53 +10:00
|
|
|
if (fd > 0) {
|
|
|
|
close(fd);
|
|
|
|
}
|
2019-12-12 01:24:40 +11:00
|
|
|
return returncode;
|
2019-12-10 14:01:22 +11:00
|
|
|
}
|
|
|
|
|