mirror of
https://github.com/fail0verflow/hbc.git
synced 2024-11-17 23:29:21 +01:00
526 lines
10 KiB
C
526 lines
10 KiB
C
/*
|
|
* Copyright (C) 2008 dhewg, #wiidev efnet
|
|
*
|
|
* this file is part of geckoloader
|
|
* http://wiibrew.org/index.php?title=Geckoloader
|
|
*
|
|
* 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 <sys/types.h>
|
|
#include <sys/stat.h>
|
|
#include <fcntl.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <unistd.h>
|
|
#include <errno.h>
|
|
#include <libgen.h>
|
|
#ifndef _WIN32
|
|
#include <sys/socket.h>
|
|
#include <netinet/in.h>
|
|
#include <arpa/inet.h>
|
|
#include <netdb.h>
|
|
#else
|
|
#include <winsock2.h>
|
|
#endif
|
|
|
|
#include <zlib.h>
|
|
|
|
#include "gecko.h"
|
|
|
|
#define WIILOAD_VERSION_MAYOR 0
|
|
#define WIILOAD_VERSION_MINOR 5
|
|
|
|
#define LD_TCP_PORT 4299
|
|
|
|
#define MAX_ARGS_LEN 1024
|
|
|
|
#ifndef O_BINARY
|
|
#define O_BINARY 0
|
|
#endif
|
|
|
|
typedef unsigned char u8;
|
|
typedef unsigned short u16;
|
|
typedef unsigned int u32;
|
|
typedef unsigned long long u64;
|
|
typedef signed char s8;
|
|
typedef signed short s16;
|
|
typedef signed int s32;
|
|
typedef signed long long s64;
|
|
|
|
typedef enum { false, true } bool;
|
|
|
|
#ifndef __WIN32__
|
|
static const char *desc_export = "export";
|
|
#ifndef __APPLE__
|
|
static const char *desc_gecko = "/dev/ttyUSB0";
|
|
#else
|
|
static const char *desc_gecko = "/dev/tty.usbserial-GECKUSB0";
|
|
#endif
|
|
#else
|
|
static const char *desc_export = "set";
|
|
static const char *desc_gecko = "COM4";
|
|
#endif
|
|
|
|
static const char *envvar = "WIILOAD";
|
|
|
|
static bool send_gecko (const char *dev, const u8 *buf, u32 len, u32 len_un,
|
|
const char *args, u16 args_len) {
|
|
u8 b[4];
|
|
u32 left, block;
|
|
const u8 *p;
|
|
|
|
if (gecko_open (dev)) {
|
|
fprintf (stderr, "unable to open the device '%s'\n", dev);
|
|
return false;
|
|
}
|
|
|
|
printf ("sending upload request\n");
|
|
|
|
b[0] = 'H';
|
|
b[1] = 'A';
|
|
b[2] = 'X';
|
|
b[3] = 'X';
|
|
|
|
if (gecko_write (b, 4)) {
|
|
gecko_close ();
|
|
fprintf (stderr, "error sending data\n");
|
|
return false;
|
|
}
|
|
|
|
b[0] = WIILOAD_VERSION_MAYOR;
|
|
b[1] = WIILOAD_VERSION_MINOR;
|
|
b[2] = (args_len >> 8) & 0xff;
|
|
b[3] = args_len & 0xff;
|
|
|
|
if (gecko_write (b, 4)) {
|
|
gecko_close ();
|
|
fprintf (stderr, "error sending data\n");
|
|
return false;
|
|
}
|
|
|
|
printf ("sending file size (%u bytes)\n", len);
|
|
|
|
b[0] = (len >> 24) & 0xff;
|
|
b[1] = (len >> 16) & 0xff;
|
|
b[2] = (len >> 8) & 0xff;
|
|
b[3] = len & 0xff;
|
|
|
|
if (gecko_write (b, 4)) {
|
|
gecko_close ();
|
|
fprintf (stderr, "error sending data\n");
|
|
return false;
|
|
}
|
|
|
|
b[0] = (len_un >> 24) & 0xff;
|
|
b[1] = (len_un >> 16) & 0xff;
|
|
b[2] = (len_un >> 8) & 0xff;
|
|
b[3] = len_un & 0xff;
|
|
|
|
if (gecko_write (b, 4)) {
|
|
gecko_close ();
|
|
fprintf (stderr, "error sending data\n");
|
|
return false;
|
|
}
|
|
|
|
printf ("sending data");
|
|
fflush (stdout);
|
|
|
|
left = len;
|
|
p = buf;
|
|
while (left) {
|
|
block = left;
|
|
if (block > 63488)
|
|
block = 63488;
|
|
left -= block;
|
|
|
|
if (gecko_write (p, block)) {
|
|
fprintf (stderr, "error sending block\n");
|
|
break;
|
|
}
|
|
p += block;
|
|
|
|
printf (".");
|
|
fflush (stdout);
|
|
|
|
}
|
|
printf ("\n");
|
|
|
|
if (args_len) {
|
|
printf ("sending arguments (%u bytes)\n", args_len);
|
|
|
|
if (gecko_write ((u8 *) args, args_len)) {
|
|
gecko_close ();
|
|
return false;
|
|
}
|
|
}
|
|
|
|
gecko_close ();
|
|
|
|
return true;
|
|
}
|
|
|
|
static bool tcp_write (int s, const u8 *buf, u32 len) {
|
|
s32 left, block;
|
|
const u8 *p;
|
|
|
|
left = len;
|
|
p = buf;
|
|
while (left) {
|
|
block = send (s, p, left, 0);
|
|
|
|
if (block < 0) {
|
|
perror ("send failed");
|
|
return false;
|
|
}
|
|
|
|
left -= block;
|
|
p += block;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
static bool send_tcp (const char *host, const u8 *buf, u32 len, u32 len_un,
|
|
const char *args, u16 args_len) {
|
|
struct sockaddr_in sa;
|
|
struct hostent *he;
|
|
int s, bc;
|
|
u8 b[4];
|
|
off_t left, block;
|
|
const u8 *p;
|
|
|
|
#ifdef __WIN32__
|
|
WSADATA wsa_data;
|
|
if (WSAStartup (MAKEWORD(2,2), &wsa_data)) {
|
|
printf ("WSAStartup failed\n");
|
|
return false;
|
|
}
|
|
#endif
|
|
|
|
memset (&sa, 0, sizeof (sa));
|
|
|
|
sa.sin_addr.s_addr = inet_addr (host);
|
|
|
|
if (sa.sin_addr.s_addr == INADDR_NONE) {
|
|
printf ("resolving %s\n", host);
|
|
|
|
he = gethostbyname (host);
|
|
|
|
if (!he) {
|
|
#ifndef __WIN32__
|
|
herror ("error resolving hostname");
|
|
#else
|
|
fprintf (stderr, "error resolving hostname\n");
|
|
#endif
|
|
return false;
|
|
}
|
|
|
|
if (he->h_addrtype != AF_INET) {
|
|
fprintf (stderr, "unsupported address");
|
|
return false;
|
|
}
|
|
|
|
sa.sin_addr.s_addr = *((u32 *) he->h_addr);
|
|
}
|
|
|
|
s = socket (PF_INET, SOCK_STREAM, 0);
|
|
|
|
if (s < 0) {
|
|
perror ("error creating socket");
|
|
return false;
|
|
}
|
|
|
|
sa.sin_port = htons (LD_TCP_PORT);
|
|
sa.sin_family = AF_INET;
|
|
|
|
printf ("connecting to %s:%d\n", inet_ntoa (sa.sin_addr), LD_TCP_PORT);
|
|
|
|
if (connect (s, (struct sockaddr *) &sa, sizeof (sa)) == -1) {
|
|
perror ("error connecting");
|
|
close (s);
|
|
return false;
|
|
}
|
|
|
|
printf ("sending upload request\n");
|
|
|
|
b[0] = 'H';
|
|
b[1] = 'A';
|
|
b[2] = 'X';
|
|
b[3] = 'X';
|
|
|
|
if (!tcp_write (s, b, 4)) {
|
|
close (s);
|
|
return false;
|
|
}
|
|
|
|
b[0] = WIILOAD_VERSION_MAYOR;
|
|
b[1] = WIILOAD_VERSION_MINOR;
|
|
b[2] = (args_len >> 8) & 0xff;
|
|
b[3] = args_len & 0xff;
|
|
|
|
if (!tcp_write (s, b, 4)) {
|
|
close (s);
|
|
return false;
|
|
}
|
|
|
|
printf ("sending file size (%u bytes)\n", len);
|
|
|
|
b[0] = (len >> 24) & 0xff;
|
|
b[1] = (len >> 16) & 0xff;
|
|
b[2] = (len >> 8) & 0xff;
|
|
b[3] = len & 0xff;
|
|
|
|
if (!tcp_write (s, b, 4)) {
|
|
close (s);
|
|
return false;
|
|
}
|
|
|
|
b[0] = (len_un >> 24) & 0xff;
|
|
b[1] = (len_un >> 16) & 0xff;
|
|
b[2] = (len_un >> 8) & 0xff;
|
|
b[3] = len_un & 0xff;
|
|
|
|
if (!tcp_write (s, b, 4)) {
|
|
close (s);
|
|
return false;
|
|
}
|
|
|
|
printf ("sending data");
|
|
fflush (stdout);
|
|
|
|
left = len;
|
|
p = buf;
|
|
bc = 0;
|
|
while (left) {
|
|
block = left;
|
|
if (block > 4 * 1024)
|
|
block = 4 * 1024;
|
|
left -= block;
|
|
|
|
if (!tcp_write (s, p, block)) {
|
|
close (s);
|
|
return false;
|
|
}
|
|
|
|
p += block;
|
|
bc++;
|
|
|
|
if (!(bc % 16)) {
|
|
printf (".");
|
|
fflush (stdout);
|
|
}
|
|
}
|
|
|
|
printf ("\n");
|
|
|
|
if (args_len) {
|
|
printf ("sending arguments (%u bytes)\n", args_len);
|
|
|
|
if (!tcp_write (s, (u8 *) args, args_len)) {
|
|
close (s);
|
|
return false;
|
|
}
|
|
}
|
|
|
|
#ifndef __WIN32__
|
|
close (s);
|
|
#else
|
|
shutdown (s, SD_SEND);
|
|
closesocket (s);
|
|
WSACleanup ();
|
|
#endif
|
|
|
|
return true;
|
|
}
|
|
|
|
static void usage (const char *argv0) {
|
|
fprintf (stderr, "set the environment variable %s to a valid "
|
|
"destination.\n\n"
|
|
"examples:\n"
|
|
"\tusbgecko mode:\n"
|
|
"\t\t%s %s=%s\n\n"
|
|
"\ttcp mode:\n"
|
|
"\t\t%s %s=tcp:wii\n"
|
|
"\t\t%s %s=tcp:192.168.0.30\n\n"
|
|
"usage:\n"
|
|
"\t%s <filename> <application arguments>\n\n",
|
|
envvar,
|
|
desc_export, envvar, desc_gecko,
|
|
desc_export, envvar,
|
|
desc_export, envvar,
|
|
argv0);
|
|
exit (EXIT_FAILURE);
|
|
}
|
|
|
|
int main (int argc, char **argv) {
|
|
int fd;
|
|
struct stat st;
|
|
char *ev;
|
|
bool compress = true;
|
|
u8 *buf, *bufz;
|
|
off_t fsize;
|
|
uLongf bufzlen = 0;
|
|
u32 len, len_un;
|
|
|
|
int i, c;
|
|
char args[MAX_ARGS_LEN];
|
|
char *arg_pos;
|
|
u16 args_len, args_left;
|
|
|
|
bool res;
|
|
|
|
printf ("wiiload v%u.%u\n"
|
|
"coded by dhewg, #wiidev efnet\n\n",
|
|
WIILOAD_VERSION_MAYOR, WIILOAD_VERSION_MINOR);
|
|
|
|
if (argc < 2)
|
|
usage (*argv);
|
|
|
|
ev = getenv (envvar);
|
|
if (!ev)
|
|
usage (*argv);
|
|
|
|
fd = open (argv[1], O_RDONLY | O_BINARY);
|
|
if (fd < 0) {
|
|
perror ("error opening the file");
|
|
exit (EXIT_FAILURE);
|
|
}
|
|
|
|
if (fstat (fd, &st)) {
|
|
close (fd);
|
|
perror ("error stat'ing the file");
|
|
exit (EXIT_FAILURE);
|
|
}
|
|
|
|
fsize = st.st_size;
|
|
|
|
if (fsize < 64 || fsize > 20 * 1024 * 1024) {
|
|
close (fd);
|
|
fprintf (stderr, "error: invalid file size\n");
|
|
exit (EXIT_FAILURE);
|
|
}
|
|
|
|
buf = malloc (fsize);
|
|
if (!buf) {
|
|
close (fd);
|
|
fprintf (stderr, "out of memory\n");
|
|
exit (EXIT_FAILURE);
|
|
}
|
|
|
|
if (read (fd, buf, fsize) != fsize) {
|
|
close (fd);
|
|
free (buf);
|
|
perror ("error reading the file");
|
|
exit (EXIT_FAILURE);
|
|
}
|
|
close (fd);
|
|
|
|
len = fsize;
|
|
len_un = 0;
|
|
|
|
if (!memcmp(buf, "PK\x03\x04", 4))
|
|
compress = false;
|
|
|
|
if (compress) {
|
|
bufzlen = (uLongf) ((float) fsize * 1.02);
|
|
|
|
bufz = malloc (bufzlen);
|
|
if (!bufz) {
|
|
fprintf (stderr, "out of memory\n");
|
|
exit (EXIT_FAILURE);
|
|
}
|
|
|
|
printf("compressing %u bytes...", (u32) fsize);
|
|
fflush(stdout);
|
|
|
|
res = compress2 (bufz, &bufzlen, buf, fsize, 6);
|
|
if (res != Z_OK) {
|
|
free(buf);
|
|
free(bufz);
|
|
fprintf (stderr, "error compressing data: %d\n", res);
|
|
exit (EXIT_FAILURE);
|
|
}
|
|
|
|
if (bufzlen < (u32) fsize) {
|
|
printf(" %.2f%%\n", 100.0f * (float) bufzlen / (float) fsize);
|
|
|
|
len = bufzlen;
|
|
len_un = fsize;
|
|
free(buf);
|
|
buf = bufz;
|
|
} else {
|
|
printf(" compressed size gained size, discarding\n");
|
|
free(bufz);
|
|
}
|
|
}
|
|
|
|
args_len = 0;
|
|
|
|
arg_pos = args;
|
|
args_left = MAX_ARGS_LEN;
|
|
|
|
c = snprintf (arg_pos, args_left, "%s", basename (argv[1]));
|
|
arg_pos += c + 1;
|
|
args_left -= c + 1;
|
|
|
|
if (argc > 2) {
|
|
for (i = 2; i < argc; ++i) {
|
|
c = snprintf (arg_pos, args_left, "%s", argv[i]);
|
|
|
|
if (c >= args_left) {
|
|
free (buf);
|
|
fprintf (stderr, "argument string too long\n");
|
|
exit (EXIT_FAILURE);
|
|
}
|
|
|
|
arg_pos += c + 1;
|
|
args_left -= c + 1;
|
|
}
|
|
|
|
if (args_left < 1) {
|
|
free (buf);
|
|
fprintf (stderr, "argument string too long\n");
|
|
exit (EXIT_FAILURE);
|
|
}
|
|
}
|
|
|
|
arg_pos[0] = 0;
|
|
args_len = MAX_ARGS_LEN - args_left + 1;
|
|
|
|
if (strncmp (ev, "tcp:", 4)) {
|
|
if (stat (ev, &st))
|
|
usage (*argv);
|
|
|
|
res = send_gecko (ev, buf, len, len_un, args, args_len);
|
|
} else {
|
|
if (strlen (ev) < 5)
|
|
usage (*argv);
|
|
|
|
res = send_tcp (&ev[4], buf, len, len_un, args, args_len);
|
|
}
|
|
|
|
if (res)
|
|
printf ("done.\n");
|
|
else
|
|
printf ("transfer failed.\n");
|
|
|
|
free (buf);
|
|
|
|
return 0;
|
|
}
|
|
|