/* * 0xFFFF - Open Free Fiasco Firmware Flasher * Copyright (C) 2007, 2008 pancake * * 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 3 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, see . */ //#include "main.h" #include #include #include #include /* global pr0n */ #if HAVE_USB #include struct usb_device *device = NULL; extern struct usb_dev_handle *dev = NULL; #endif //char *fiasco_image = NULL; char *boot_cmdline = NULL; char *reverseto = NULL; char *subverstr = NULL; int rd_mode = -1; int rd_flags = -1; int usb_mode = -1; int root_device = -1; int simulate = 0; int noverify = 0; int verbose = 0; int identify = 0; int moboreboot = 0; //int unpack = 0; int qmode = 0; int info = 0; int nomode = 0; /*char *modes[]={ "host", "peripheral", NULL };*/ char *root_devices[] = { "flash", "mmc", "usb", NULL }; void show_title() { printf("0xFFFF v%s // The Free Fiasco Firmware Flasher\n", VERSION); } void show_usage() { int i; show_title(); #if 0 // TODO: remove console: serial.c // TODO: remove inject usb and supported list // TODO: remove option for calculate hash printf ("" #if defined(WITH_USB) && ! defined(WITH_DEVICE) "Over USB:\n" " -b [cmdline] boot default or loaded kernel with cmdline (default: use default kernel cmdline)\n" " -R reboot device\n" " -l load all specified images to RAM\n" " -f flash all specified images\n" " -c cold flash 2nd and secondary images\n" "\n" #endif #ifdef WITH_DEVICE "On device:\n" " -R reboot device\n" " -f flash all specified images\n" " -x [/dev/mtd] check for bad blocks on mtd device (default: check all)\n" " -E dump all images from device to fiasco image, see -M and -t\n" " -e [dir] dump all images from device to directory, see -t (default: current dir)\n" "\n" #endif #if defined(WITH_USB) || defined(WITH_DEVICE) "Device configuration:\n" " -i identify, show all information about device\n" " -D 0|1|2 change root device: 0 - flash, 1 - mmc, 2 - usb\n" " -U 0|1 disable/enable USB host mode\n" " -r 0|1 disable/enable R&D mode\n" " -F flags change R&D flags, flags are comma separated list, can be empty\n" " -H rev change HW revision\n" " -K ver change kernel version string\n" " -N ver change NOLO version string\n" " -S ver change SW release version string\n" " -C ver change content eMMC version string\n" "\n" #endif "Normal image specification:\n" " -m [arg] specify normal image\n" " arg is [[[dev:[hw:]]ver:]type:]file[%%lay]\n" " dev is device name string (default: emtpy)\n" " hw are comma separated list of HW revisions (default: empty)\n" " ver is image version string (default: empty)\n" " type is image type (default: try autodetect)\n" " file is image file name\n" " lay is layout file name (default: none)\n" "\n" "Fiasco image specification:\n" " -M file specify input fiasco image\n" " -t [types] filter images by comma separated list of image types (default: do not filter)\n" " -d [dev] filter images by device (default: do not filter)\n" " -w [hw] filter images by HW revision (default: do not filter)\n" " -u [dir] unpack fiasco image to directory (default: current dir)\n" " -g [ver] generate fiasco image for SW release version (default: without SW release)\n" "\n" "Other options:\n" " -I identify images\n" #if defined(WITH_USB) || defined(WITH_DEVICE) " -c console prompt mode\n" #endif #if ( defined(WITH_USB) || defined(WITH_DEVICE) ) && defined(WITH_SQUEUES) " -Q enter shared queues server mode (for gui or remote)\n" #endif " -s simulate, do not flash or write on disk\n" " -n disable hash, checksum and image type checking\n" " -v be verbose and noisy\n" " -V show 0xFFFF version information\n" " -h show this help message\n" "\n" #if defined(WITH_USB) || defined(WITH_DEVICE) "R&D flags:\n" " no-omap-wd disable auto reboot by OMAP watchdog\n" " no-ext-wd disable auto reboot by external watchdog\n" " no-lifeguard-reset disable auto reboot by software lifeguard\n" " serial-console enable serial console\n" " no-usb-timeout disable usb timeout for flashing\n" " sti-console enable sti console\n" " no-charging disable battery charging\n" " force-power-key force omap boot reason to power key\n" "\n" #endif ); printf( "Devices:\n"); for ( i = 0; i < DEVICE_COUNT; ++i ) if ( device_to_string(i) && device_to_long_string(i) ) printf(" %-14s %s\n", device_to_string(i), device_to_long_string(i)); printf( "\n"); printf( "Image types:\n"); for ( i = 0; i < IMAGE_COUNT; ++i ) if ( image_type_to_string(i) ) printf(" %s\n", image_type_to_string(i)); printf( "\n"); #endif #if HAVE_USB printf ("Over USB:\n" " -b [arg] boots the kernel with arguments\n" " -c console prompt mode\n" " -d [vid:pid] injects a usb device into the supported list\n" " -D [0|1|2] sets the root device to flash (0), mmc (1) or usb (2)\n" " -f set the given RD flags (see '-f help')\n" " -i show device information (let standby mode)\n" " -l list supported usb device ids\n" " -p format piece-of-firmware\n" " piece format: [[[[dev:[hw:]]ver:]type:]file[%%layout]\n" " hw is device HW revision\n" " dev is device name string\n" " ver is image version string\n" " type is image type\n" " file is image file name\n" " layout is file name for layout file\n" " -r [0|1] disable/enable R&D mode\n" " -R reboot the omap board\n" " -U [0|1] disable/enable the usb host mode\n"); #endif #if HAVE_SQUEUE printf(" -Q enter shared queues server mode (for gui or remote)\n"); #endif printf ("Local stuff:\n" // " -s [serial] serial port console (minicom like terminal)\n" " -h show this help message\n" " -S [subversion] unpacks/flash pieces matching this sub-version information\n" " -n do not flash or write to disk (simulation)\n" " -C [/dev/mtd] check bad blocks on mtd\n" " -e [path] dump/extract pieces to path\n" // " -F [fiasco] flash a fiasco firmware image\n" " -H [file] calculate hash for file\n" " -I [piece] identify a firmware piece\n" // " -P [new-fiasco] creates a new fiasco package, pieces as arguments\n" // " optional argument 'version:'\n" // " -u [fiasco] unpack target fiasco image\n" " -v be verbose and noisy\n" " -V show 0xFFFF version information\n" " -x extract configuration entries from /dev/mtd1\n" "Pieces are: "); // for(i=0;pieces[i];i++) printf("%s ", pieces[i]); printf("\n"); // serial port support is not yet done (cold flash is for flashing the 8kB nand) // TODO: commandline shell prompt for nolo comm exit(0); } /*int unpack_callback(struct header_t *header) { FILE *fd; fiasco_data_read(header); if (subverstr) { if (strchr(header->name, ',') != NULL) { if (!strstr(header->name, subverstr)) { printf("Skipping '%s' does not matches -S subversion\n", header->name); return 1; } } } if (nomode) return 1; printf("Unpacking file: %s\n", header->name); printf(" type: %s\n", header->type); if (header->device[0]) printf(" device: %s\n", header->device); if (header->hwrevs[0]) printf(" hw revisions: %s\n", header->hwrevs); if (header->version[0]) printf(" version: %s\n", header->version); if (header->layout) printf(" layout file: %s\n", header->layout); printf(" size: %d\n", header->size); printf(" hash: %04x\n", header->hash); fd = fopen(header->name, "wb"); if (fd == NULL) { printf("Cannot open file.\n"); return 1; } if (fwrite(header->data, header->size, 1, fd) != 1) printf("Writing failed\n"); fclose(fd); return 0; } int flash_callback(struct header_t *header) { int ret; char *type; ret = unpack_callback(header); if (ret) { printf("ignored\n"); return 1; } type = header->type; if (!type[0]) type = (char *)fpid_file(header->name); printf("\nFlashing %s (%s)\n", header->name, type); flash_image(header->name, type, header->device, header->hwrevs, header->version); return 0; } void unpack_fiasco_image(char *file) { printf("Dumping firmware pieces to disk.\n"); fiasco_callback = &unpack_callback; openfiasco(file, NULL, NULL, NULL, NULL, 0); } int fiasco_flash(const char *file) { char *p; char version[64]; if (connect_via_usb()) { fprintf(stderr, "Cannot connect to device. It is possibly not in boot stage.\n"); return 0; } // if (info) cmd_info(""); check_nolo_order(); get_nolo_version(); get_sw_version(); get_hw_revision(version, 44); if (subverstr == NULL) { p = strstr(version, "hw_rev:"); if (p) { subverstr = strdup(p+7); // TODO: delimit string by comma printf("SubVersionString autodetected: '%s'\n", subverstr); } } printf("\n"); fiasco_callback = &flash_callback; openfiasco(file, "xloader", NULL, NULL, NULL, 0); openfiasco(file, "secondary", NULL, NULL, NULL, 0); openfiasco(file, "kernel", NULL, NULL, NULL, 0); openfiasco(file, "initfs", NULL, NULL, NULL, 0); openfiasco(file, "rootfs", NULL, NULL, NULL, 0); return 0; }*/ #if HAVE_USB int connect_via_usb() { static char pbc[]={'/','-','\\', '|'}; struct usb_device_descriptor udd; // struct devices it_device; int c = 0; // usb_set_debug(5); usb_init(); /* Tries to get access to the Internet Tablet and retries * if any of the neccessary steps fail. * * Note: While a proper device may be found on the bus it may * not be in the right state to be accessed (e.g. the Nokia is * not in the boot stage any more). */ while(!dev) { usleep(0xc350); // 0.5s /* if(!usb_device_found(&udd, &it_device)) { printf("\rWaiting for device... %c", pbc[++c%4]); fflush(stdout); continue; }*/ /* open device */ if(!(dev = usb_open(device))) { perror("usb_open"); return 1; } #if 1 { char name[32]; name[0] = '\0'; usb_get_driver_np(dev, 2, name, sizeof(name)); printf ("Driver: %s\n", name); usb_detach_kernel_driver_np(dev, 2); } #endif if (usb_claim_interface(dev, 2) < 0) { //device->config->interface->altsetting->bInterfaceNumber) < 0) { // 2 or 0 perror("usb_claim_interface"); // Something is broken if closing fails. if(usb_close(dev)) { perror("usb_close"); return 1; } dev = NULL; // Try again later.? exit(1); } if (usb_set_altinterface(dev, 1) <0) { //device->config->interface->altsetting->bAlternateSetting) < 0) { perror("usb_set_altinterface"); // Something is broken if closing fails. if(usb_close(dev)) { perror("usb_close"); return 1; } dev = NULL; // Try again later. continue; } break; } // printf("found %s (%04x:%04x)\n", it_device.name, // it_device.vendor_id, it_device.product_id); /* go go go! */ while(get_status()); sleep(1); // take breath return 0; } #endif int main(int argc, char **argv) { int c; const char *type; while((c = getopt(argc, argv, "QC:cp:PvVhRu:ib:U:r:e:ld:I:D:f:F:s:xH:S:n")) != -1) { switch(c) { // case 'H': // printf("xorpair: %04x\n", do_hash_file(optarg, NULL)); // return 0; case 'x': return dump_config(); #if HAVE_USB case 'c': return console_prompt(); case 'b': boot_cmdline = optarg; break; case 'S': subverstr = optarg; break; case 'n': nomode = 1; break; case 'U': usb_mode = atoi(optarg); break; // case 'F': // fiasco_image = optarg; // break; // case 'd': // sscanf(optarg, "%04hx:%04hx", // &supported_devices[SUPPORTED_DEVICES-2].vendor_id, // &supported_devices[SUPPORTED_DEVICES-2].product_id); // supported_devices[SUPPORTED_DEVICES-2].name = strdup("user"); // break; case 'D': root_device = atoi(optarg); break; case 'f': if (!strcmp(optarg,"help")) { printf("* Flags are composed of:\n"); printf(" 0x02 - disable OMAP watchdog (possibly)\n"); printf(" 0x04 - disable RETU watchdog (possibly)\n"); printf(" 0x08 - disable lifeguard reset\n"); printf(" 0x10 - enable serial console\n"); printf(" 0x20 - disable USB timeout\n"); exit(1); } rd_flags = (unsigned short) strtoul(optarg, NULL, 16); break; case 'r': rd_mode = atoi(optarg); break; // case 'l': // list_valid_devices(); // return 0; // case 'p': // add_piece(optarg); // break; case 'i': info = 1; break; case 'R': moboreboot = 1; break; #endif case 'e': reverseto = optarg; break; // case 's': // return console(optarg); // case 'u': // fiasco_image = optarg; // unpack = 1; // break; case 'Q': qmode = 1; break; // case 'P': // return fiasco_pack(optind, argv); // case 'I': // type = fpid_file(optarg); // printf("%s: %s\n", type, optarg); // if (type && strcmp(type, "fiasco") == 0) // openfiasco(optarg, NULL, NULL, NULL, NULL, 1); // identify = 1; // break; case 'C': return check_badblocks(optarg); case 'v': verbose = 1; break; case 'h': show_usage(); break; case 'V': printf("%s\n", VERSION); return 0; } } if (qmode) return queue_mode(); // if (!unpack && fiasco_image) // return fiasco_flash(fiasco_image); if (identify) return 0; // flags ok? if ( /*(fiasco_image == NULL)*/ 1 && (boot_cmdline == NULL) && (reverseto == NULL) // && (pcs_n == 0) && (rd_flags == -1) && (rd_mode == -1) && (info == 0) && (moboreboot == 0) && (usb_mode == -1) && (root_device == -1)) { printf("# The Free Fiasco Firmware Flasher v"VERSION"\n" "0xFFFF [-chinlQRvVx] [-C mtd-dev] [-d vid:pid] [-D 0|1|2] [-e path] [-f flags]\n" " [-F fiasco] [-H hash-file] [-I piece] [-p [piece%%]file]] [-r 0|1] [-S subver]\n" " [-s serial-dev] [-u fiasco-image] [-U 0|1] | [-P new-fiasco] [piece1] [2] ..\n"); return 1; } /* if (unpack) { if (reverseto) { if (chdir(reverseto) < 0) { printf("Error: Cannot change directory to %s\n", reverseto); return 1; } } unpack_fiasco_image(fiasco_image); return 0; }*/ if (reverseto) { reverse_extract_pieces(reverseto); return 0; } #if HAVE_USB if (connect_via_usb()) { fprintf(stderr, "Cannot connect to device. It is possibly not in boot stage.\n"); return 0; } // if (info) // cmd_info(""); /* if (pcs_n) { char version[64]; // check_nolo_order(); get_sw_version(); get_hw_revision(version, 44); if (subverstr == NULL) { char *p = strstr(version, "hw_rev:"); if (p) { subverstr = strdup(p+7); // TODO: delimit string by comma printf("SubVersionString autodetected: '%s'\n", subverstr); } } get_nolo_version(); for(c=0;c