add scripts used to upload binaries to the keyboard

This commit is contained in:
Daniel Zanco 2022-09-16 00:13:47 -03:00
parent 7c523a7718
commit a9b8d553fa
3 changed files with 255 additions and 0 deletions

View File

@ -0,0 +1,59 @@
#!/usr/bin/python
import time
import serial
import serial.tools.list_ports as lports
import argparse
def main(args):
# Find which serial port to use (assumes only the keyboard is connected)
if args.port == '':
port_list = lports.comports()
if len(port_list) != 1:
print('No serial ports detected, quitting...')
quit()
else:
port = port_list[0].device
else:
port = args.port
# Based on https://github.com/rogerclarkmelbourne/Arduino_STM32/blob/master/tools/linux/src/upload-reset/upload-reset.c
# Send magic sequence of DTR and RTS followed by the magic word "1EAF"
with serial.Serial(port=port, baudrate=args.baudrate) as ser:
ser.rts = False
ser.dtr = False
ser.dtr = True
time.sleep(0.05)
ser.dtr = False
ser.rts = True
ser.dtr = True
time.sleep(0.05)
ser.dtr = False
time.sleep(0.05)
ser.write(b'1EAF')
# Wait for the DFU mode reboot
time.sleep(args.sleep * 1e-3)
if __name__ == '__main__':
parser = argparse.ArgumentParser()
parser = argparse.ArgumentParser(description="Enter DFU mode of DevTerm's keyboard.")
parser.add_argument('-p', '--port', dest='port', type=str, default='',
help='Serial port (default: detect serial ports).')
parser.add_argument('-b', '--baudrate', dest='baudrate', type=int, default=9600,
help='Baudrate (default: 9600bps).')
parser.add_argument('-s', '--sleep', dest='sleep', type=int, default=1500,
help='Wait SLEEP milliseconds (ms) after enabling DFU mode (default: 1500ms).')
args = parser.parse_args()
main(args)

View File

@ -0,0 +1,166 @@
/* Copyright (C) 2015 Roger Clark <www.rogerclark.net>
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
*
* Utility to send the reset sequence on RTS and DTR and chars
* which resets the libmaple and causes the bootloader to be run
*
*
*
* Terminal control code by Heiko Noordhof (see copyright below)
*/
/* Copyright (C) 2003 Heiko Noordhof <heikyAusers.sf.net>
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*/
#include <stdio.h>
#include <stdlib.h>
#include <termios.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <sys/ioctl.h>
#include <stdbool.h>
/* Function prototypes (belong in a seperate header file) */
int openserial(char *devicename);
void closeserial(void);
int setDTR(unsigned short level);
int setRTS(unsigned short level);
/* Two globals for use by this module only */
static int fd;
static struct termios oldterminfo;
void closeserial(void)
{
tcsetattr(fd, TCSANOW, &oldterminfo);
close(fd);
}
int openserial(char *devicename)
{
struct termios attr;
if ((fd = open(devicename, O_RDWR)) == -1)
return 0; /* Error */
atexit(closeserial);
if (tcgetattr(fd, &oldterminfo) == -1)
return 0; /* Error */
attr = oldterminfo;
attr.c_cflag |= CRTSCTS | CLOCAL;
attr.c_oflag = 0;
if (tcflush(fd, TCIOFLUSH) == -1)
return 0; /* Error */
if (tcsetattr(fd, TCSANOW, &attr) == -1)
return 0; /* Error */
/* Set the lines to a known state, and */
/* finally return non-zero is successful. */
return setRTS(0) && setDTR(0);
}
/* For the two functions below:
* level=0 to set line to LOW
* level=1 to set line to HIGH
*/
int setRTS(unsigned short level)
{
int status;
if (ioctl(fd, TIOCMGET, &status) == -1)
{
perror("setRTS(): TIOCMGET");
return 0;
}
if (level)
status |= TIOCM_RTS;
else
status &= ~TIOCM_RTS;
if (ioctl(fd, TIOCMSET, &status) == -1)
{
perror("setRTS(): TIOCMSET");
return 0;
}
return 1;
}
int setDTR(unsigned short level)
{
int status;
if (ioctl(fd, TIOCMGET, &status) == -1)
{
perror("setDTR(): TIOCMGET");
return 0;
}
if (level)
status |= TIOCM_DTR;
else
status &= ~TIOCM_DTR;
if (ioctl(fd, TIOCMSET, &status) == -1)
{
perror("setDTR: TIOCMSET");
return 0;
}
return 1;
}
/* This portion of code was written by Roger Clark
* It was informed by various other pieces of code written by Leaflabs to reset their
* Maple and Maple mini boards
*/
main(int argc, char *argv[])
{
if (argc < 2 || argc > 3)
{
printf("Usage upload-reset <serial_device> <Optional_delay_in_milliseconds>\n\r");
return;
}
if (openserial(argv[1]))
{
// Send magic sequence of DTR and RTS followed by the magic word "1EAF"
setRTS(false);
setDTR(false);
setDTR(true);
usleep(50000L);
setDTR(false);
setRTS(true);
setDTR(true);
usleep(50000L);
setDTR(false);
usleep(50000L);
write(fd, "1EAF", 4);
closeserial();
if (argc == 3)
{
usleep(atol(argv[2]) * 1000L);
}
}
else
{
printf("Failed to open serial device.\n\r");
}
}

View File

@ -0,0 +1,30 @@
#!/bin/bash
altID="2"
usbID="1EAF:0003"
serial_port="/dev/ttyACM0"
# Default export location
# binfile="../devterm_keyboard.ino.generic_stm32f103r8.bin"
binfile="devterm_keyboard.ino.bin"
# Send magic numbers via serial to enter DFU mode.
# This needs pyserial, so it's installed quietly:
# pip install -q pyserial
./dfumode.py -p $serial_port -b 9600 -s 1500
# Alternatively you can compile the C program that does the same thing:
# gcc upload-reset.c -o upload-reset
# ./upload-reset $serial_port 1500
# Upload binary file
dfu-util -d ${usbID} -a ${altID} -D ${binfile} -R
echo "Waiting for $serial_port serial..."
COUNTER=0
while [ ! -c $serial_port ] && ((COUNTER++ < 40)); do
sleep 0.1
done
echo Done