/* * This file is part of the flashrom project. * * Copyright (C) 2012 Virgil-Adrian Teaca * * 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; version 2 of the License. * * 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., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA */ /* Driver for the SI-Prog like hardware by Lancos.com. * See http://www.lancos.com/siprogsch.html for schematics and instructions. */ #include #include #include "flash.h" #include "programmer.h" /* We have: MOSI -----> DTR MISO -----> CTS SCK --+--> RTS +--> DSR CS# -----> TXD */ enum pony_type { TYPE_SI_PROG, TYPE_SERBANG, }; /* The hardware programmer used. */ static enum pony_type pony_type = TYPE_SI_PROG; static void pony_bitbang_set_cs(int val) { if (pony_type == TYPE_SI_PROG) { /* CS# is negated by the SI-Prog programmer. */ val ^= 1; } sp_set_pin(PIN_TXD, val); } static void pony_bitbang_set_sck(int val) { sp_set_pin(PIN_RTS, val); } static void pony_bitbang_set_mosi(int val) { sp_set_pin(PIN_DTR, val); } static int pony_bitbang_get_miso(void) { int tmp; tmp = sp_get_pin(PIN_CTS); if (pony_type == TYPE_SERBANG) { /* MISO is negated by the SERBANG programmer. */ tmp ^= 1; } return tmp; } static const struct bitbang_spi_master bitbang_spi_master_pony = { .type = BITBANG_SPI_MASTER_PONY, .set_cs = pony_bitbang_set_cs, .set_sck = pony_bitbang_set_sck, .set_mosi = pony_bitbang_set_mosi, .get_miso = pony_bitbang_get_miso, .half_period = 0, }; int pony_spi_init(void) { char *arg = NULL; int i, data_in, data_out, have_device = 0; int have_prog = 1; /* The parameter is on format "dev=/dev/device,type=serbang" */ arg = extract_programmer_param("dev"); if (arg && strlen(arg)) { sp_fd = sp_openserport( arg, 9600 ); if (sp_fd < 0) { free(arg); return 1; } have_device++; } free(arg); if (!have_device) { msg_perr("Error: No valid device specified.\n" "Use flashrom -p pony_spi:dev=/dev/device\n"); return 1; } /* Register the shutdown callback. */ if (register_shutdown(serialport_shutdown, NULL)) { return 1; } arg = extract_programmer_param("type"); if (arg) { if (!strcasecmp( arg, "serbang")) { pony_type = TYPE_SERBANG; } else if (!strcasecmp( arg, "si_prog") ) { pony_type = TYPE_SI_PROG; } else { msg_perr("Error: Invalid programmer type specified.\n"); free(arg); return 1; } } free(arg); msg_pdbg("Using the %s programmer.\n", ((pony_type == TYPE_SI_PROG ) ? "SI-Prog" : "SERBANG")); /* * Detect if there is a SI-Prog compatible programmer connected. */ pony_bitbang_set_cs(1); pony_bitbang_set_mosi(1); /* We toggle SCK while we keep MOSI and CS# on. */ for (i = 1; i <= 10; ++i) { data_out = i & 1; sp_set_pin(PIN_RTS, data_out); programmer_delay( 1000 ); data_in = sp_get_pin(PIN_DSR); if (data_out != data_in) { have_prog = 0; break; } } if (!have_prog) { msg_perr( "No SI-Prog compatible hardware detected.\n" ); return 1; } if (bitbang_spi_init(&bitbang_spi_master_pony)) { return 1; } return 0; }