mirror of
https://github.com/fail0verflow/bootmii-utils.git
synced 2024-11-29 05:34:13 +01:00
200 lines
4.0 KiB
C
200 lines
4.0 KiB
C
|
/*
|
||
|
* Copyright (C) 2008 dhewg, #wiidev efnet
|
||
|
*
|
||
|
* this file is part of wiifuse
|
||
|
* http://wiibrew.org/index.php?title=Wiifuse
|
||
|
*
|
||
|
* 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.
|
||
|
*
|
||
|
* This program is distributed in the hope that it will be useful,
|
||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||
|
* GNU General Public License for more details.
|
||
|
*
|
||
|
* You should have received a copy of the GNU General Public License
|
||
|
* along with this program; if not, write to the Free Software
|
||
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||
|
*/
|
||
|
|
||
|
#include <stdio.h>
|
||
|
#include <string.h>
|
||
|
#include <unistd.h>
|
||
|
#include <fcntl.h>
|
||
|
#include <errno.h>
|
||
|
|
||
|
#ifndef __WIN32__
|
||
|
#include <termios.h>
|
||
|
#else
|
||
|
#define WIN32_LEAN_AND_MEAN
|
||
|
#include <windows.h>
|
||
|
#endif
|
||
|
|
||
|
#include "gecko.h"
|
||
|
|
||
|
#define FTDI_PACKET_SIZE 3968
|
||
|
|
||
|
#ifndef __WIN32__
|
||
|
static int fd_gecko = -1;
|
||
|
#else
|
||
|
static HANDLE handle_gecko = NULL;
|
||
|
#endif
|
||
|
|
||
|
int gecko_open(const char *dev) {
|
||
|
#ifndef __WIN32__
|
||
|
struct termios newtio;
|
||
|
|
||
|
fd_gecko = open(dev, O_RDWR | O_NOCTTY | O_SYNC | O_NONBLOCK);
|
||
|
|
||
|
if (fd_gecko == -1) {
|
||
|
perror("gecko_open");
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
if (fcntl(fd_gecko, F_SETFL, 0)) {
|
||
|
perror("F_SETFL on serial port");
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
if (tcgetattr(fd_gecko, &newtio)) {
|
||
|
perror("tcgetattr");
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
cfmakeraw(&newtio);
|
||
|
|
||
|
newtio.c_cflag |= CRTSCTS | CS8 | CLOCAL | CREAD;
|
||
|
|
||
|
if (tcsetattr(fd_gecko, TCSANOW, &newtio)) {
|
||
|
perror("tcsetattr");
|
||
|
return 1;
|
||
|
}
|
||
|
#else
|
||
|
COMMTIMEOUTS timeouts;
|
||
|
|
||
|
handle_gecko = CreateFile(dev, GENERIC_READ | GENERIC_WRITE, 0, 0,
|
||
|
OPEN_EXISTING, 0, 0);
|
||
|
|
||
|
GetCommTimeouts(handle_gecko, &timeouts);
|
||
|
|
||
|
timeouts.ReadIntervalTimeout = MAXDWORD;
|
||
|
timeouts.ReadTotalTimeoutMultiplier = 0;
|
||
|
timeouts.ReadTotalTimeoutConstant = 0;
|
||
|
timeouts.WriteTotalTimeoutMultiplier = 0;
|
||
|
timeouts.WriteTotalTimeoutConstant = 0;
|
||
|
|
||
|
if (!SetCommTimeouts(handle_gecko, &timeouts)) {
|
||
|
fprintf(stderr, "error setting timeouts on port\n");
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
if (!SetCommMask(handle_gecko, 0)) {
|
||
|
fprintf(stderr, "error setting communications event mask\n");
|
||
|
return 1;
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
gecko_flush();
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
void gecko_close() {
|
||
|
#ifndef __WIN32__
|
||
|
if (fd_gecko > 0)
|
||
|
close(fd_gecko);
|
||
|
#else
|
||
|
CloseHandle(handle_gecko);
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
void gecko_flush() {
|
||
|
#ifndef __WIN32__
|
||
|
tcflush(fd_gecko, TCIOFLUSH);
|
||
|
#else
|
||
|
PurgeComm(handle_gecko, PURGE_RXCLEAR | PURGE_TXCLEAR |
|
||
|
PURGE_TXABORT | PURGE_RXABORT);
|
||
|
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
int gecko_read(void *buf, size_t count) {
|
||
|
size_t left, chunk;
|
||
|
#ifndef __WIN32__
|
||
|
size_t res;
|
||
|
#else
|
||
|
DWORD res;
|
||
|
#endif
|
||
|
|
||
|
left = count;
|
||
|
while (left) {
|
||
|
chunk = left;
|
||
|
if (chunk > FTDI_PACKET_SIZE)
|
||
|
chunk = FTDI_PACKET_SIZE;
|
||
|
|
||
|
#ifndef __WIN32__
|
||
|
res = read(fd_gecko, buf, chunk);
|
||
|
if (res < 1) {
|
||
|
perror("gecko_read");
|
||
|
return 1;
|
||
|
}
|
||
|
#else
|
||
|
if (!ReadFile(handle_gecko, buf, chunk, &res, NULL)) {
|
||
|
fprintf(stderr, "gecko_read\n");
|
||
|
return 1;
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
left -= res;
|
||
|
buf += res;
|
||
|
}
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
int gecko_write(const void *buf, size_t count) {
|
||
|
size_t left, chunk;
|
||
|
#ifndef __WIN32__
|
||
|
size_t res;
|
||
|
#else
|
||
|
DWORD res;
|
||
|
#endif
|
||
|
|
||
|
left = count;
|
||
|
|
||
|
while (left) {
|
||
|
chunk = left;
|
||
|
if (chunk > FTDI_PACKET_SIZE)
|
||
|
chunk = FTDI_PACKET_SIZE;
|
||
|
|
||
|
#ifndef __WIN32__
|
||
|
res = write(fd_gecko, buf, count);
|
||
|
if (res < 1) {
|
||
|
perror("gecko_write");
|
||
|
return 1;
|
||
|
}
|
||
|
#else
|
||
|
if (!WriteFile(handle_gecko, buf, chunk, &res, NULL)) {
|
||
|
fprintf (stderr, "gecko_write\n");
|
||
|
return 1;
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
left -= res;
|
||
|
buf += res;
|
||
|
|
||
|
#ifndef __WIN32__
|
||
|
// does this work with ftdi-sio?
|
||
|
if (tcdrain(fd_gecko)) {
|
||
|
perror ("gecko_drain");
|
||
|
return 1;
|
||
|
}
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|