| |
| /* |
| * common.c - common wrapper functions between barebox and the host |
| * |
| * Copyright (c) 2007 Sascha Hauer <s.hauer@pengutronix.de>, Pengutronix |
| * |
| * See file CREDITS for list of people who contributed to this |
| * project. |
| * |
| * This program is free software; you can redistribute it and/or modify |
| * it under the terms of the GNU General Public License version 2 |
| * as published by the Free Software Foundation. |
| * |
| * 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 |
| */ |
| |
| /** |
| * @file |
| * @brief Common wrapper functions between barebox and the host |
| */ |
| /* |
| * These are host includes. Never include any barebox header |
| * files here... |
| */ |
| #include <stdio.h> |
| #include <stdlib.h> |
| #include <stdint.h> |
| #include <termios.h> |
| #include <unistd.h> |
| #include <fcntl.h> |
| #include <time.h> |
| #include <getopt.h> |
| #include <sys/types.h> |
| #include <sys/stat.h> |
| #include <string.h> |
| #include <libgen.h> |
| #include <sys/mman.h> |
| #include <errno.h> |
| #include <signal.h> |
| #include <sys/select.h> |
| /* |
| * ...except the ones needed to connect with barebox |
| */ |
| #include <mach/linux.h> |
| #include <mach/hostfile.h> |
| |
| static struct termios term_orig, term_vi; |
| static char erase_char; /* the users erase character */ |
| |
| static void rawmode(void) |
| { |
| tcgetattr(0, &term_orig); |
| term_vi = term_orig; |
| term_vi.c_lflag &= (~ICANON & ~ECHO & ~ISIG); |
| term_vi.c_iflag &= (~IXON & ~ICRNL); |
| term_vi.c_oflag |= (ONLCR); |
| term_vi.c_cc[VMIN] = 1; |
| term_vi.c_cc[VTIME] = 0; |
| erase_char = term_vi.c_cc[VERASE]; |
| tcsetattr(0, TCSANOW, &term_vi); |
| } |
| |
| static void cookmode(void) |
| { |
| fflush(stdout); |
| tcsetattr(0, TCSANOW, &term_orig); |
| } |
| |
| int linux_tstc(int fd) |
| { |
| struct timeval tv = { |
| .tv_usec = 100, |
| }; |
| fd_set rfds; |
| int ret; |
| |
| FD_ZERO(&rfds); |
| FD_SET(fd, &rfds); |
| |
| /* |
| * We set the timeout here to 100us, because otherwise |
| * barebox would eat all cpu resources while waiting |
| * for input. |
| */ |
| ret = select(fd + 1, &rfds, NULL, NULL, &tv); |
| |
| if (ret) |
| return 1; |
| |
| return 0; |
| } |
| |
| int ctrlc(void) |
| { |
| char chr; |
| |
| if (linux_read_nonblock(0, &chr, 1) == 1 && chr == 3) |
| return 1; |
| return 0; |
| } |
| |
| uint64_t linux_get_time(void) |
| { |
| struct timespec ts; |
| uint64_t now; |
| |
| clock_gettime(CLOCK_MONOTONIC, &ts); |
| |
| now = ts.tv_sec * 1000 * 1000 * 1000 + ts.tv_nsec; |
| |
| return now; |
| } |
| |
| void __attribute__((noreturn)) reset_cpu(unsigned long addr) |
| { |
| cookmode(); |
| exit(0); |
| } |
| |
| void enable_interrupts(void) |
| { |
| } |
| |
| void disable_interrupt(void) |
| { |
| } |
| |
| int linux_read(int fd, void *buf, size_t count) |
| { |
| ssize_t ret; |
| |
| if (count == 0) |
| return 0; |
| |
| do { |
| ret = read(fd, buf, count); |
| |
| if (ret == 0) { |
| printf("read on fd %d returned 0, device gone? - exiting\n", fd); |
| reset_cpu(0); |
| } else if (ret == -1) { |
| if (errno == EAGAIN) |
| return -errno; |
| else if (errno == EINTR) |
| continue; |
| else { |
| printf("read on fd %d returned -1, errno %d - exiting\n", fd, errno); |
| reset_cpu(0); |
| } |
| } |
| } while (ret <= 0); |
| |
| return (int)ret; |
| } |
| |
| int linux_read_nonblock(int fd, void *buf, size_t count) |
| { |
| int oldflags, ret; |
| |
| oldflags = fcntl(fd, F_GETFL); |
| if (oldflags == -1) |
| goto err_out; |
| |
| if (fcntl(fd, F_SETFL, oldflags | O_NONBLOCK) == -1) |
| goto err_out; |
| |
| ret = linux_read(fd, buf, count); |
| |
| if (fcntl(fd, F_SETFL, oldflags) == -1) |
| goto err_out; |
| |
| return ret; |
| |
| err_out: |
| perror("fcntl"); |
| return -1; |
| } |
| |
| ssize_t linux_write(int fd, const void *buf, size_t count) |
| { |
| return write(fd, buf, count); |
| } |
| |
| off_t linux_lseek(int fd, off_t offset) |
| { |
| return lseek(fd, offset, SEEK_SET); |
| } |
| |
| extern void start_barebox(void); |
| extern void mem_malloc_init(void *start, void *end); |
| |
| static int add_image(char *str, char *name) |
| { |
| char *file; |
| int readonly = 0, map = 1; |
| struct stat s; |
| char *opt; |
| int fd, ret; |
| struct hf_platform_data *hf = malloc(sizeof(struct hf_platform_data)); |
| |
| if (!hf) |
| return -1; |
| |
| file = strtok(str, ","); |
| while ((opt = strtok(NULL, ","))) { |
| if (!strcmp(opt, "ro")) |
| readonly = 1; |
| if (!strcmp(opt, "map")) |
| map = 1; |
| } |
| |
| printf("add file %s(%s,%s)\n", file, readonly ? "ro" : "", map ? "map" : ""); |
| |
| fd = open(file, readonly ? O_RDONLY : O_RDWR); |
| hf->fd = fd; |
| hf->filename = file; |
| |
| if (fd < 0) { |
| perror("open"); |
| goto err_out; |
| } |
| |
| if (fstat(fd, &s)) { |
| perror("fstat"); |
| goto err_out; |
| } |
| |
| hf->size = s.st_size; |
| hf->name = strdup(name); |
| |
| if (map) { |
| hf->map_base = (unsigned long)mmap(NULL, hf->size, |
| PROT_READ | (readonly ? 0 : PROT_WRITE), |
| MAP_SHARED, fd, 0); |
| if ((void *)hf->map_base == MAP_FAILED) |
| printf("warning: mmapping %s failed\n", file); |
| } |
| |
| ret = barebox_register_filedev(hf); |
| if (ret) |
| goto err_out; |
| return 0; |
| |
| err_out: |
| if (fd > 0) |
| close(fd); |
| free(hf); |
| return -1; |
| } |
| |
| static void print_usage(const char*); |
| |
| int main(int argc, char *argv[]) |
| { |
| void *ram; |
| int opt, ret, fd; |
| int malloc_size = 8 * 1024 * 1024; |
| char str[6]; |
| int fdno = 0, envno = 0; |
| |
| ram = malloc(malloc_size); |
| if (!ram) { |
| printf("unable to get malloc space\n"); |
| exit(1); |
| } |
| mem_malloc_init(ram, ram + malloc_size); |
| |
| while (1) { |
| int option_index = 0; |
| static struct option long_options[] = { |
| {"help", 0, 0, 'h'}, |
| {"image", 1, 0, 'i'}, |
| {"env", 1, 0, 'e'}, |
| {"stdout", 1, 0, 'O'}, |
| {"stdin", 1, 0, 'I'}, |
| {0, 0, 0, 0}, |
| }; |
| |
| opt = getopt_long(argc, argv, "hi:e:O:I:", |
| long_options, &option_index); |
| |
| if (opt == -1) |
| break; |
| |
| switch (opt) { |
| case 'h': |
| print_usage(basename(argv[0])); |
| exit(0); |
| case 'i': |
| sprintf(str, "fd%d", fdno); |
| ret = add_image(optarg, str); |
| if (ret) |
| exit(1); |
| fdno++; |
| break; |
| case 'm': |
| /* This option is broken. add_image needs malloc, so |
| * mem_alloc_init() has to be called before option |
| * parsing |
| */ |
| malloc_size = strtoul(optarg, NULL, 0); |
| break; |
| case 'e': |
| sprintf(str, "env%d", envno); |
| ret = add_image(optarg, str); |
| if (ret) |
| exit(1); |
| envno++; |
| break; |
| case 'O': |
| fd = open(optarg, O_WRONLY); |
| if (fd < 0) { |
| perror("open"); |
| exit(1); |
| } |
| |
| barebox_register_console("cout", -1, fd); |
| break; |
| case 'I': |
| fd = open(optarg, O_RDWR); |
| if (fd < 0) { |
| perror("open"); |
| exit(1); |
| } |
| |
| barebox_register_console("cin", fd, -1); |
| break; |
| default: |
| exit(1); |
| } |
| } |
| |
| barebox_register_console("console", fileno(stdin), fileno(stdout)); |
| |
| rawmode(); |
| start_barebox(); |
| |
| /* never reached */ |
| return 0; |
| } |
| |
| #ifdef __PPC__ |
| /* HACK: we need this symbol on PPC, better ask a PPC export about this :) */ |
| char _SDA_BASE_[4096]; |
| #endif |
| |
| static void print_usage(const char *prgname) |
| { |
| printf( |
| "Usage: %s [OPTIONS]\n" |
| "Start barebox.\n\n" |
| "Options:\n\n" |
| " -i, --image=<file> Map an image file to barebox. This option can be given\n" |
| " multiple times. The files will show up as\n" |
| " /dev/fd0 ... /dev/fdx under barebox.\n" |
| " -e, --env=<file> Map a file with an environment to barebox. With this \n" |
| " option, files are mapped as /dev/env0 ... /dev/envx\n" |
| " and thus are used as the default environment.\n" |
| " An empty file generated with dd will do to get started\n" |
| " with an empty environment.\n" |
| " -O, --stdout=<file> Register a file as a console capable of doing stdout.\n" |
| " <file> can be a regular file or a FIFO.\n" |
| " -I, --stdin=<file> Register a file as a console capable of doing stdin.\n" |
| " <file> can be a regular file or a FIFO.\n", |
| prgname |
| ); |
| } |
| |
| /** |
| * @page barebox_simul barebox Simulator |
| * |
| * barebox can be run as a simulator on your host to check and debug new non |
| * hardware related features. |
| * |
| * @section simu_build How to build barebox for simulation |
| * |
| * @section simu_run How to run barebox simulator |
| * |
| * $ barebox [\<OPTIONS\>] |
| * |
| * Options can be: |
| * |
| * -i \<file\> |
| * |
| * Map a \<file\> to barebox. This option can be given multiple times. The \<file\>s |
| * will show up as /dev/fd0 ... /dev/fdx in the barebox simulator. |
| * |
| * -e \<file\> |
| * |
| * Map \<file\> to barebox. With this option \<file\>s are mapped as /dev/env0 ... |
| * /dev/envx and thus are used as default environment. A clean file generated |
| * with dd will do to get started with an empty environment |
| * |
| * -O \<file\> |
| * |
| * Register \<file\> as a console capable of doing stdout. \<file\> can be a |
| * regular file or a fifo. |
| * |
| * -I \<file\> |
| * |
| * Register \<file\> as a console capable of doing stdin. \<file\> can be a regular |
| * file or a fifo. |
| * |
| * @section simu_dbg How to debug barebox simulator |
| * |
| */ |