123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067 |
- /*
- * Copyright (C) 2011 Google, Inc.
- * Copyright (C) 2010 Samsung Electronics.
- *
- * This software is licensed under the terms of the GNU General Public
- * License version 2, as published by the Free Software Foundation, and
- * may be copied, distributed, and modified under those terms.
- *
- * 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.
- *
- */
- #include <linux/irq.h>
- #include <linux/gpio.h>
- #include <linux/time.h>
- #include <linux/interrupt.h>
- #include <linux/timer.h>
- #include <linux/wakelock.h>
- #include <linux/delay.h>
- #include <linux/wait.h>
- #include <linux/sched.h>
- #include <linux/vmalloc.h>
- #include <linux/if_arp.h>
- #include <linux/platform_device.h>
- #include <linux/spi/spi.h>
- #include <linux/kthread.h>
- #include <linux/workqueue.h>
- #include <linux/module.h>
- #include <linux/platform_data/modem.h>
- #include <linux/sec_param.h>
- #include <linux/of_gpio.h>
- #include <mach/gpiomux.h>
- //#include <linux/platform_data/modem_v2.h>
- #include "modem_prj.h"
- #include "modem_link_device_spi.h"
- #include "modem_utils.h"
- #define USE_SPI_HALF_DUPLEX
- /* For function which has void parmeter */
- static struct spi_link_device *p_spild;
- static struct spi_device *p_spi;
- static int spi_init_ipc(struct spi_link_device *spild);
- #ifdef CONFIG_LINK_DEVICE_SPI_DEBUG
- void spi_print_data(char *buf, int len)
- {
- int words = len >> 4;
- int residue = len - (words << 4);
- int i;
- char *b;
- char last[80];
- char tb[8];
- /* Make the last line, if ((len % 16) > 0) */
- if (residue > 0) {
- memset(last, 0, sizeof(last));
- memset(tb, 0, sizeof(tb));
- b = buf + (words << 4);
- sprintf(last, "%04X: ", (words << 4));
- for (i = 0; i < residue; i++) {
- sprintf(tb, "%02x ", b[i]);
- strcat(last, tb);
- if ((i & 0x3) == 0x3) {
- sprintf(tb, " ");
- strcat(last, tb);
- }
- }
- }
- for (i = 0; i < words; i++) {
- b = buf + (i << 4);
- mif_err("%04X: "
- "%02x %02x %02x %02x %02x %02x %02x %02x "
- "%02x %02x %02x %02x %02x %02x %02x %02x\n",
- (i << 4),
- b[0], b[1], b[2], b[3], b[4], b[5], b[6], b[7],
- b[8], b[9], b[10], b[11], b[12], b[13], b[14], b[15]);
- }
- /* Print the last line */
- if (residue > 0)
- mif_err("%s\n", last);
- }
- #endif
- static void spi_send_work(int spi_sigs, bool spi_work_t)
- {
- struct spi_work_type *spi_wq;
- spi_wq = kmalloc(sizeof(struct spi_work_type), GFP_ATOMIC);
- if (unlikely(!spi_wq)) {
- pr_err("[LNK/E] <%s> Failed to kmalloc()\n", __func__);
- return;
- }
- spi_wq->signal_code = spi_sigs;
- INIT_WORK(&spi_wq->work, spi_work);
- // GSCHO
- if (spi_work_t == SPI_WORK_FRONT)
- queue_work_front(p_spild->spi_wq, (struct work_struct *)spi_wq);
- else
- queue_work(p_spild->spi_wq, (struct work_struct *)spi_wq);
- }
- static irqreturn_t spi_cp_dump_irq_handler(int irq, void *p_ld)
- {
- struct link_device *ld = (struct link_device *)p_ld;
- struct spi_link_device *spild = to_spi_link_device(ld);
- irqreturn_t result = IRQ_HANDLED;
- #ifdef CONFIG_LINK_DEVICE_SPI_DEBUG
- pr_info("[SPI] [%s] cp_dump_int irq received. \n",__func__);
- #endif
- if (!spild->boot_done)
- goto exit;
- if (!wake_lock_active(&spild->spi_wake_lock)
- && spild->send_modem_spi != 1) {
- wake_lock(&spild->spi_wake_lock);
- pr_debug("[SPI] [%s](%d) spi_wakelock locked . spild->spi_state[%d]\n",
- __func__, __LINE__, (int)spild->spi_state);
- }
- exit:
- return result;
- }
- static irqreturn_t spi_srdy_irq_handler(int irq, void *p_ld)
- {
- struct link_device *ld = (struct link_device *)p_ld;
- struct spi_link_device *spild = to_spi_link_device(ld);
- struct io_device *iod;
- irqreturn_t result = IRQ_HANDLED;
- #ifdef CONFIG_LINK_DEVICE_SPI_DEBUG
- pr_info("[SPI] [%s](%d) irq received. spild->spi_state[%d]\n",
- __func__, __LINE__, (int)spild->spi_state);
- #endif
- if (!spild->boot_done)
- goto exit1;
- if (!wake_lock_active(&spild->spi_wake_lock)
- && spild->send_modem_spi != 1) {
- wake_lock(&spild->spi_wake_lock);
- pr_debug("[SPI] [%s](%d) spi_wakelock locked . spild->spi_state[%d]\n",
- __func__, __LINE__, (int)spild->spi_state);
- }
- if (spild->send_modem_spi == 1)
- up(&spild->srdy_sem);
- /* SRDY interrupt work on SPI_STATE_IDLE state for receive data */
- if (spild->spi_state == SPI_STATE_IDLE
- || spild->spi_state == SPI_STATE_RX_TERMINATE
- || spild->spi_state == SPI_STATE_TX_TERMINATE) {
- iod = link_get_iod_with_format(&spild->ld, IPC_FMT);
- if (!iod) {
- mif_err("no iodevice for modem control\n");
- goto exit;
- }
- if (iod->mc->phone_state != STATE_ONLINE)
- goto exit;
- #ifdef CONFIG_LINK_DEVICE_SPI_DEBUG
- pr_err("[SPI] [%s](%d)call spi_send_work spild->spi_state[%d]\n",
- __func__, __LINE__, (int)spild->spi_state);
- #endif
- spi_send_work(SPI_WORK_RECEIVE, SPI_WORK_FRONT);
- }
- return result;
- exit:
- if (wake_lock_active(&spild->spi_wake_lock)) {
- wake_unlock(&spild->spi_wake_lock);
- pr_debug("[SPI] [%s](%d) spi_wakelock unlocked.\n",
- __func__, __LINE__);
- }
- exit1:
- return result;
- }
- static irqreturn_t spi_subsrdy_irq_handler(int irq, void *p_ld)
- {
- struct link_device *ld = (struct link_device *)p_ld;
- struct spi_link_device *spild = to_spi_link_device(ld);
- irqreturn_t result = IRQ_HANDLED;
- /* SRDY interrupt work on SPI_STATE_TX_WAIT state for send data */
- if (spild->spi_state == SPI_STATE_TX_WAIT)
- return result;
- pr_debug("%s spild->spi_state[%d]\n",
- "[SPI] spi_main_subsrdy_rising_handler :",
- (int)spild->spi_state);
- return result;
- }
- static int spi_send
- (
- struct link_device *ld,
- struct io_device *iod,
- struct sk_buff *skb
- )
- {
- struct sk_buff_head *txq;
- enum dev_format fmt = iod->format;
- const u32 cmd_ready = 0x12341234;
- const u32 cmd_start = 0x45674567;
- int ret;
- u32 data;
- switch (fmt) {
- case IPC_FMT:
- case IPC_RAW:
- case IPC_RFS:
- txq = p_spild->skb_txq[fmt];
- skb_queue_tail(txq, skb);
- ret = skb->len;
- break;
- case IPC_BOOT:
- if (get_user(data, (u32 __user *)skb->data))
- return -EFAULT;
- if (data == cmd_ready) {
- p_spild->ril_send_modem_img = 1;
- p_spild->ril_send_cnt = 0;
- } else if (data == cmd_start) {
- p_spild->ril_send_modem_img = 0;
- if (!queue_work(p_spild->spi_modem_wq,
- &p_spild->send_modem_w))
- pr_err("(%d) already exist w-q\n",
- __LINE__);
- } else {
- if (p_spild->ril_send_modem_img) {
- memcpy((void *)(p_spild->p_virtual_buff +
- p_spild->ril_send_cnt),
- skb->data, skb->len);
- p_spild->ril_send_cnt += skb->len;
- }
- }
- ret = skb->len;
- dev_kfree_skb_any(skb);
- return ret;
- default:
- pr_err("[LNK/E] <%s:%s> No TXQ for %s\n",
- __func__, ld->name, iod->name);
- dev_kfree_skb_any(skb);
- return 0;
- }
- spi_send_work(SPI_WORK_SEND, SPI_WORK);
- return ret;
- }
- //SPI_SETUP
- static int spi_check_cp_bin(struct spi_link_device *spild)
- {
- /* need to dowload : 1, don't need : 0 */
- int retval = 0;
- sec_get_param(param_update_cp_bin, (void*)&retval);
- return retval;
- }
- static int spi_updated_cp_bin(struct spi_link_device *spild)
- {
- /* don't need to download ZZ: 0 */
- int retval = 0, updated_cp_bin=0;
- sec_set_param(param_update_cp_bin, (void*)&updated_cp_bin);
- return retval;
- }
- static int spi_ioctl(struct link_device *ld, struct io_device *iod,
- unsigned int cmd, unsigned long arg)
- {
- struct spi_link_device *spild = to_spi_link_device(ld);
- int err = 0;
- mif_debug("%s: spi_ioctl cmd 0x%08X\n", ld->name, cmd);
- switch (cmd) {
- case IOCTL_DPRAM_CHECK_CP_BIN:
- //modified. dklee
- spi_init_ipc(p_spild);
- err = spi_check_cp_bin(spild);
- mif_debug("%s: check_cp_bin [%d]\n", ld->name, err);
- break;
- case IOCTL_DPRAM_UPDATED_CP_BIN:
- err = spi_updated_cp_bin(spild);
- mif_debug("%s: check_cp_bin [%d]\n", ld->name, err);
- break;
- default:
- pr_err("%s: ERR! invalid cmd 0x%08X\n", ld->name, cmd);
- err = -EINVAL;
- break;
- }
- return err;
- }
- static int spi_register_isr
- (
- unsigned irq,
- irqreturn_t (*isr)(int, void*),
- unsigned long flag,
- const char *name,
- struct link_device *ld
- )
- {
- int ret = 0;
- ret = request_irq(irq, isr, flag, name, ld);
- if (ret) {
- pr_err("[LNK/E] <%s> request_irq fail (%d)\n",
- __func__, ret);
- goto err;
- }
- if (irq == gpio_to_irq(p_spild->gpio_cp_dump_int)){
- ret = enable_irq_wake(irq);
- if (ret) {
- pr_err("[LNK/E] <%s> enable_irq_wake fail (%d)\n",
- __func__, ret);
- free_irq(irq, ld);
- goto err;
- }
- }
- pr_debug("[LNK] <%s> IRQ#%d handler is registered.\n", __func__, irq);
- err:
- return ret;
- }
- void spi_unregister_isr(unsigned irq, void *data)
- {
- free_irq(irq, data);
- }
- int spi_tx_rx_sync(u8 *tx_d, u8 *rx_d, unsigned len)
- {
- struct spi_transfer t;
- struct spi_message msg;
- memset(&t, 0, sizeof t);
- t.len = len;
- t.tx_buf = tx_d;
- t.rx_buf = rx_d;
- t.cs_change = 1;
- t.bits_per_word = 32;
- //t.bits_per_word = 8;
- // t.speed_hz = 10800000;
- #ifdef CONFIG_LINK_DEVICE_SPI_DEBUG
- printk("%s : len :- %d\n", __func__, len);
- printk("%s : mode :- %d\n", __func__, p_spi->mode);
- #endif
- spi_message_init(&msg);
- spi_message_add_tail(&t, &msg);
- return spi_sync(p_spi, &msg);
- }
- static int spi_buff_write
- (
- struct spi_link_device *spild,
- int dev_id,
- const char *buff,
- unsigned size
- )
- {
- u32 templength, buf_length;
- u32 spi_data_mux;
- u32 spi_packet_free_length;
- u8 *spi_packet;
- struct spi_data_packet_header *spi_packet_header;
- spi_packet_header = (struct spi_data_packet_header *)spild->buff;
- spi_packet = (char *)spild->buff;
- spi_packet_free_length = SPI_DATA_PACKET_MAX_PACKET_BODY_SIZE -
- spi_packet_header->current_data_size;
- buf_length = size + SPI_DATA_MUX_SIZE + SPI_DATA_LENGTH_SIZE;
- /* not enough space in spi packet */
- if (spi_packet_free_length < buf_length) {
- spi_packet_header->more = 1;
- return 0;
- }
- /* check spi mux type */
- switch (dev_id) {
- case IPC_FMT:
- spi_data_mux = SPI_DATA_MUX_IPC;
- break;
- case IPC_RAW:
- spi_data_mux = SPI_DATA_MUX_RAW;
- break;
- case IPC_RFS:
- spi_data_mux = SPI_DATA_MUX_RFS;
- break;
- default:
- pr_err("%s %s\n",
- "[SPI] ERROR : spi_buff_write:",
- "invalid dev_id");
- return 0;
- }
- /* copy spi mux field */
- memcpy(spi_packet + SPI_DATA_PACKET_HEADER_SIZE +
- spi_packet_header->current_data_size,
- &spi_data_mux, SPI_DATA_MUX_SIZE);
- spi_packet_header->current_data_size += SPI_DATA_MUX_SIZE;
- /* copy spi data length field */
- templength = size-SPI_DATA_BOF_SIZE-SPI_DATA_EOF_SIZE;
- memcpy(spi_packet + SPI_DATA_PACKET_HEADER_SIZE +
- spi_packet_header->current_data_size,
- &templength, SPI_DATA_LENGTH_SIZE);
- spi_packet_header->current_data_size += SPI_DATA_LENGTH_SIZE;
- /* copy data field */
- memcpy(spi_packet + SPI_DATA_PACKET_HEADER_SIZE +
- spi_packet_header->current_data_size,
- buff, size);
- spi_packet_header->current_data_size += size;
- return buf_length;
- }
- static void spi_prepare_tx_packet(void)
- {
- struct sk_buff *skb;
- int ret;
- int i;
- for (i = 0; i < p_spild->max_ipc_dev; i++) {
- while ((skb = skb_dequeue(p_spild->skb_txq[i]))) {
- ret = spi_buff_write(p_spild, i, skb->data, skb->len);
- if (!ret) {
- skb_queue_head(p_spild->skb_txq[i], skb);
- break;
- }
- dev_kfree_skb_any(skb);
- }
- }
- }
- static void spi_start_data_send(void)
- {
- int i;
- for (i = 0; i < p_spild->max_ipc_dev; i++) {
- if (skb_queue_len(p_spild->skb_txq[i]) > 0)
- spi_send_work(SPI_WORK_SEND, SPI_WORK);
- }
- }
- static void spi_tx_work(void)
- {
- struct spi_link_device *spild;
- struct io_device *iod;
- char *spi_packet_buf;
- char *spi_sync_buf;
- spild = p_spild;
- iod = link_get_iod_with_format(&spild->ld, IPC_FMT);
- if (!iod) {
- mif_err("no iodevice for modem control\n");
- return;
- }
- if (iod->mc->phone_state != STATE_ONLINE)
- return;
- /* check SUB SRDY, SRDY state */
- if (gpio_get_value(spild->gpio_ipc_sub_srdy) ==
- SPI_GPIOLEVEL_HIGH ||
- gpio_get_value(spild->gpio_ipc_srdy) ==
- SPI_GPIOLEVEL_HIGH) {
- spi_start_data_send();
- return;
- }
- // GSCHO
- if (get_console_suspended())
- return;
- if (spild->spi_state == SPI_STATE_END)
- return;
- /* change state SPI_STATE_IDLE to SPI_STATE_TX_WAIT */
- spild->spi_state = SPI_STATE_TX_WAIT;
- spild->spi_timer_tx_state = SPI_STATE_TIME_START;
- gpio_direction_output(spild->gpio_ipc_mrdy, SPI_GPIOLEVEL_HIGH);
- /* Start TX timer */
- spild->spi_tx_timer.expires = jiffies +
- ((SPI_TIMER_TX_WAIT_TIME * HZ) / 1000);
- add_timer(&spild->spi_tx_timer);
- /* check SUBSRDY state */
- while (gpio_get_value(spild->gpio_ipc_sub_srdy) ==
- SPI_GPIOLEVEL_LOW) {
- if (spild->spi_timer_tx_state == SPI_STATE_TIME_OVER) {
- pr_err("%s spild->spi_state=[%d]\n",
- "[spi_tx_work] == spi Fail to receiving SUBSRDY CONF :",
- (int)spild->spi_state);
- spild->spi_timer_tx_state = SPI_STATE_TIME_START;
- gpio_direction_output(spild->gpio_ipc_mrdy,
- SPI_GPIOLEVEL_LOW);
- /* change state SPI_STATE_TX_WAIT */
- /* to SPI_STATE_IDLE */
- spild->spi_state = SPI_STATE_IDLE;
- spi_send_work(SPI_WORK_SEND, SPI_WORK);
- return;
- }
- }
- /* Stop TX timer */
- del_timer(&spild->spi_tx_timer);
- if (spild->spi_state != SPI_STATE_START
- && spild->spi_state != SPI_STATE_END
- && spild->spi_state != SPI_STATE_INIT) {
- spi_packet_buf = spild->buff;
- spi_sync_buf = spild->sync_buff;
- gpio_direction_output(spild->gpio_ipc_sub_mrdy, SPI_GPIOLEVEL_HIGH);
- gpio_direction_output(spild->gpio_ipc_mrdy, SPI_GPIOLEVEL_LOW);
- /* change state SPI_MAIN_STATE_TX_SENDING */
- spild->spi_state = SPI_STATE_TX_SENDING;
- memset(spi_packet_buf, 0, SPI_MAX_PACKET_SIZE);
- memset(spi_sync_buf, 0, SPI_MAX_PACKET_SIZE);
- spi_prepare_tx_packet();
- #ifdef USE_SPI_HALF_DUPLEX
- if (spi_tx_rx_sync((void *)spi_packet_buf, (void *)NULL,
- SPI_MAX_PACKET_SIZE)) {
- #else
- if (spi_tx_rx_sync((void *)spi_packet_buf, (void *)spi_sync_buf,
- SPI_MAX_PACKET_SIZE)) {
- #endif
- /* TODO: save failed packet */
- /* back data to each queue */
- pr_err("[SPI] spi_dev_send fail\n");
- /* add cp reset when spi sync fail */
- if (iod)
- iod->modem_state_changed(iod,
- STATE_CRASH_RESET);
- }
- spild->spi_state = SPI_STATE_TX_TERMINATE;
- gpio_direction_output(spild->gpio_ipc_sub_mrdy, SPI_GPIOLEVEL_LOW);
- #ifdef CONFIG_LINK_DEVICE_SPI_DEBUG
- pr_info("[SPI] spi_tx_work : success - spi_state=[%d]\n",
- (int)spild->spi_state);
- #endif
- /* change state SPI_MAIN_STATE_TX_SENDING to SPI_STATE_IDLE */
- spild->spi_state = SPI_STATE_IDLE;
- spi_start_data_send();
- } else
- return;
- }
- int spi_buff_read(struct spi_link_device *spild)
- {
- struct link_device *ld;
- struct spi_data_packet_header *spi_packet_header;
- struct sk_buff *skb;
- char *spi_packet;
- int dev_id;
- unsigned int spi_packet_length;
- unsigned int spi_packet_cur_pos = SPI_DATA_PACKET_HEADER_SIZE;
- unsigned int spi_data_mux;
- unsigned int spi_data_length;
- unsigned int data_length;
- u8 *spi_cur_data;
- u8 *dst;
- spi_packet = spild->buff;
- ld = &spild->ld;
- /* check spi packet header */
- if (*(unsigned int *)spi_packet == 0x00000000
- || *(unsigned int *)spi_packet == 0xFFFFFFFF) {
- /* if spi header is invalid, */
- /* read spi header again with next 4 byte */
- spi_packet += SPI_DATA_PACKET_HEADER_SIZE;
- }
- /* read spi packet header */
- spi_packet_header = (struct spi_data_packet_header *)spi_packet;
- spi_packet_length = SPI_DATA_PACKET_HEADER_SIZE +
- spi_packet_header->current_data_size;
- do {
- /* read spi data mux and set current queue */
- memcpy(&spi_data_mux,
- spi_packet + spi_packet_cur_pos, SPI_DATA_MUX_SIZE);
- switch (spi_data_mux & SPI_DATA_MUX_NORMAL_MASK) {
- case SPI_DATA_MUX_IPC:
- dev_id = IPC_FMT;
- break;
- case SPI_DATA_MUX_RAW:
- dev_id = IPC_RAW;
- break;
- case SPI_DATA_MUX_RFS:
- dev_id = IPC_RFS;
- break;
- default:
- pr_err("%s len[%u], pos[%u]\n",
- "[SPI] ERROR : spi_buff_read : MUX error",
- spi_packet_length, spi_packet_cur_pos);
- return spi_packet_cur_pos - SPI_DATA_PACKET_HEADER_SIZE;
- }
- /* read spi data length */
- memcpy(&spi_data_length, spi_packet +
- spi_packet_cur_pos + SPI_DATA_LENGTH_OFFSET,
- SPI_DATA_LENGTH_SIZE);
- data_length = spi_data_length +
- SPI_DATA_BOF_SIZE + SPI_DATA_EOF_SIZE;
- spi_data_length += SPI_DATA_HEADER_SIZE;
- /* read data and make spi data */
- spi_cur_data = spi_packet + spi_packet_cur_pos;
- /* enqueue spi data */
- skb = alloc_skb(data_length, GFP_KERNEL);
- if (unlikely(!skb)) {
- pr_err("%s %s\n",
- "[SPI] ERROR : spi_buff_read:",
- "Can't allocate memory for SPI");
- return -ENOMEM;
- }
- dst = skb_put(skb, data_length);
- memcpy(dst, spi_packet +
- spi_packet_cur_pos + SPI_DATA_BOF_OFFSET,
- data_length);
- skb_queue_tail(&spild->skb_rxq[dev_id], skb);
- /* move spi packet current posision */
- spi_packet_cur_pos += spi_data_length;
- } while ((spi_packet_length - 1) > spi_packet_cur_pos);
- return 1;
- }
- static void spi_rx_work(void)
- {
- struct link_device *ld;
- struct spi_link_device *spild;
- struct sk_buff *skb;
- struct io_device *iod;
- char *spi_packet_buf;
- char *spi_sync_buf;
- int i;
- #if 0 //dklee temp
- char test_spi_send_buff[256];
- int k;
- #endif
- spild = p_spild;
- ld = &spild->ld;
- if (!spild)
- pr_err("[LNK/E] <%s> dpld == NULL\n", __func__);
- iod = link_get_iod_with_format(&spild->ld, IPC_FMT);
- if (!iod) {
- mif_err("no iodevice for modem control\n");
- return;
- }
- if (iod->mc->phone_state != STATE_ONLINE)
- return;
- if (!wake_lock_active(&spild->spi_wake_lock) ||
- gpio_get_value(spild->gpio_ipc_srdy) == SPI_GPIOLEVEL_LOW ||
- //G9009D incoming call screen delay issue
- //get_console_suspended() ||
- spild->spi_state == SPI_STATE_END)
- return;
- spild->spi_state = SPI_STATE_RX_WAIT;
- spild->spi_timer_rx_state = SPI_STATE_TIME_START;
- #if 1 //temp set gpio ourput enable
- //modem may set this pin as input mode . fix it later.
- gpio_direction_output(spild->gpio_ipc_sub_mrdy, SPI_GPIOLEVEL_HIGH);
- #else
- gpio_set_value(spild->gpio_ipc_sub_mrdy, SPI_GPIOLEVEL_HIGH);
- #endif
- /* Start TX timer */
- spild->spi_rx_timer.expires = jiffies +
- ((SPI_TIMER_RX_WAIT_TIME * HZ) / 1000);
- add_timer(&spild->spi_rx_timer);
- /* check SUBSRDY state */
- while (gpio_get_value(spild->gpio_ipc_sub_srdy) ==
- SPI_GPIOLEVEL_LOW) {
- if (spild->spi_timer_rx_state == SPI_STATE_TIME_OVER) {
- pr_err("[SPI] ERROR(Failed MASTER RX:%d ms)",
- SPI_TIMER_RX_WAIT_TIME);
- spild->spi_timer_rx_state = SPI_STATE_TIME_START;
- gpio_direction_output(spild->gpio_ipc_sub_mrdy,
- SPI_GPIOLEVEL_LOW);
- /* change state SPI_MAIN_STATE_RX_WAIT */
- /* to SPI_STATE_IDLE */
- spild->spi_state = SPI_STATE_IDLE;
- return;
- }
- }
- /* Stop TX timer */
- del_timer(&spild->spi_rx_timer);
- if (spild->spi_state == SPI_STATE_START
- || spild->spi_state == SPI_STATE_END
- || spild->spi_state == SPI_STATE_INIT)
- return;
- spi_packet_buf = spild->buff;
- spi_sync_buf = spild->sync_buff;
- memset(spi_packet_buf, 0, SPI_MAX_PACKET_SIZE);
- memset(spi_sync_buf, 0, SPI_MAX_PACKET_SIZE);
- #if 0 //DKLee test tem log
- for(k=0; k<256; k++)
- test_spi_send_buff[k]=k;
- mif_err("Data sent\n");
- spi_print_data(test_spi_send_buff, 64);
- #endif
- #if 1 //test dklee
- #ifdef USE_SPI_HALF_DUPLEX
- if (spi_tx_rx_sync((void *)NULL, (void *)spi_packet_buf,
- SPI_MAX_PACKET_SIZE) == 0) {
- #else
- if (spi_tx_rx_sync((void *)spi_sync_buf, (void *)spi_packet_buf,
- SPI_MAX_PACKET_SIZE) == 0) {
- #endif
- #else
- if (spi_tx_rx_sync((void *)test_spi_send_buff, (void *)spi_packet_buf,
- SPI_MAX_PACKET_SIZE) == 0) {
- #endif
- #ifdef CONFIG_LINK_DEVICE_SPI_DEBUG //DKLee test tem log
- mif_err("Data from CP\n");
- spi_print_data(spi_packet_buf, 64);
- #endif
- /* parsing SPI packet */
- if (spi_buff_read(spild) > 0) {
- /* call function for send data to IPC, RAW, RFS */
- for (i = 0; i < spild->max_ipc_dev; i++) {
- iod = spild->iod[i];
- while ((skb = skb_dequeue(&spild->skb_rxq[i]))
- != NULL) {
- if (iod->recv(iod, ld, skb->data,
- skb->len) < 0)
- pr_err("[LNK/E] <%s:%s> recv fail\n",
- __func__, ld->name);
- dev_kfree_skb_any(skb);
- }
- }
- }
- } else {
- pr_err("%s %s\n",
- "[SPI] ERROR : spi_rx_work :",
- "spi sync failed");
- /* add cp reset when spi sync fail */
- if (iod)
- iod->modem_state_changed(iod,
- STATE_CRASH_RESET);
- }
- spild->spi_state = SPI_STATE_RX_TERMINATE;
- gpio_direction_output(spild->gpio_ipc_sub_mrdy, SPI_GPIOLEVEL_LOW);
- /* change state SPI_MAIN_STATE_RX_WAIT to SPI_STATE_IDLE */
- spild->spi_state = SPI_STATE_IDLE;
- spi_start_data_send();
- #ifdef CONFIG_LINK_DEVICE_SPI_DEBUG //DKLee test tem log
- mif_err("Data from CP END\n");
- spi_print_data(spi_packet_buf, 64);
- #endif
-
- }
- static int spi_init_ipc(struct spi_link_device *spild)
- {
- struct link_device *ld = &spild->ld;
- int i;
- /* Make aliases to each IO device */
- for (i = 0; i < MAX_DEV_FORMAT; i++)
- spild->iod[i] = link_get_iod_with_format(ld, i);
- spild->iod[IPC_RAW] = spild->iod[IPC_MULTI_RAW];
- /* List up the IO devices connected to each IPC channel */
- for (i = 0; i < MAX_DEV_FORMAT; i++) {
- if (spild->iod[i])
- pr_err("[LNK] <%s:%s> spild->iod[%d]->name = %s\n",
- __func__, ld->name, i, spild->iod[i]->name);
- else
- pr_err("[LNK] <%s:%s> No spild->iod[%d]\n",
- __func__, ld->name, i);
- }
- return 0;
- }
- unsigned int sprd_crc_calc(char *buf_ptr, unsigned int len)
- {
- unsigned int i;
- unsigned short crc = 0;
- while (len-- != 0) {
- for (i = 0x80; i != 0 ; i = i>>1) {
- if ((crc & 0x8000) != 0) {
- crc = crc << 1 ;
- crc = crc ^ 0x1021;
- } else {
- crc = crc << 1 ;
- }
- if ((*buf_ptr & i) != 0)
- crc = crc ^ 0x1021;
- }
- buf_ptr++;
- }
- return crc;
- }
- unsigned short sprd_crc_calc_fdl(unsigned short *src, int len)
- {
- unsigned int sum = 0;
- unsigned short SourceValue, DestValue = 0;
- unsigned short lowSourceValue, hiSourceValue = 0;
- /* Get sum value of the source.*/
- while (len > 1) {
- SourceValue = *src++;
- DestValue = 0;
- lowSourceValue = (SourceValue & 0xFF00) >> 8;
- hiSourceValue = (SourceValue & 0x00FF) << 8;
- DestValue = lowSourceValue | hiSourceValue;
- sum += DestValue;
- len -= 2;
- }
- if (len == 1)
- sum += *((unsigned char *) src);
- sum = (sum >> 16) + (sum & 0x0FFFF);
- sum += (sum >> 16);
- return ~sum;
- }
- int encode_msg(struct sprd_image_buf *img, int bcrc)
- {
- u16 crc; /* CRC value*/
- u8 *src_ptr; /* source buffer pointer*/
- int dest_len; /* output buffer length*/
- u8 *dest_ptr; /* dest buffer pointer*/
- u8 high_crc, low_crc = 0;
- register int curr;
- /* CRC Check. */
- src_ptr = img->tx_b;
- /* CRC Check. */
- if (bcrc)
- crc = sprd_crc_calc(src_ptr, img->tx_size);
- else
- crc = sprd_crc_calc_fdl
- ((unsigned short *)src_ptr, img->tx_size);
- high_crc = (crc>>8) & 0xFF;
- low_crc = crc & 0xFF;
- /* Get the total size to be allocated.*/
- dest_len = 0;
- for (curr = 0; curr < img->tx_size; curr++) {
- switch (*(src_ptr+curr)) {
- case HDLC_FLAG:
- case HDLC_ESCAPE:
- dest_len += 2;
- break;
- default:
- dest_len++;
- break;
- }
- }
- switch (low_crc) {
- case HDLC_FLAG:
- case HDLC_ESCAPE:
- dest_len += 2;
- break;
- default:
- dest_len++;
- }
- switch (high_crc) {
- case HDLC_FLAG:
- case HDLC_ESCAPE:
- dest_len += 2;
- break;
- default:
- dest_len++;
- }
- dest_ptr = kmalloc(dest_len + 2, GFP_KERNEL);
- /* Memory Allocate fail.*/
- if (dest_ptr == NULL)
- return -ENOMEM;
- *dest_ptr = HDLC_FLAG;
- dest_len = 1;
- /* do escape*/
- for (curr = 0; curr < img->tx_size; curr++) {
- switch (*(src_ptr+curr)) {
- case HDLC_FLAG:
- case HDLC_ESCAPE:
- *(dest_ptr + dest_len++) = HDLC_ESCAPE;
- *(dest_ptr + dest_len++) =
- *(src_ptr + curr) ^ HDLC_ESCAPE_MASK;
- break;
- default:
- *(dest_ptr + dest_len++) = *(src_ptr + curr);
- break;
- }
- }
- switch (high_crc) {
- case HDLC_FLAG:
- case HDLC_ESCAPE:
- *(dest_ptr + dest_len++) = HDLC_ESCAPE;
- *(dest_ptr + dest_len++) = high_crc ^ HDLC_ESCAPE_MASK;
- break;
- default:
- *(dest_ptr + dest_len++) = high_crc;
- }
- switch (low_crc) {
- case HDLC_FLAG:
- case HDLC_ESCAPE:
- *(dest_ptr + dest_len++) = HDLC_ESCAPE;
- *(dest_ptr + dest_len++) = low_crc ^ HDLC_ESCAPE_MASK;
- break;
- default:
- *(dest_ptr + dest_len++) = low_crc;
- }
- *(dest_ptr + dest_len++) = HDLC_FLAG;
- memcpy(img->encoded_tx_b, dest_ptr, dest_len);
- img->encoded_tx_size = dest_len;
- kfree(dest_ptr);
- return 0;
- }
- int decode_msg(struct sprd_image_buf *img, int bcrc)
- {
- u16 crc; /* CRC value*/
- u8 *src_ptr; /* source buffer pointer*/
- int dest_len; /* output buffer length*/
- u8 *dest_ptr; /* dest buffer pointer*/
- register int curr;
- /* Check if exist End Flag.*/
- src_ptr = img->rx_b;
- dest_len = 0;
- if (img->rx_size < 4)
- return -EINVAL;
- /* Get the total size to be allocated for decoded message.*/
- for (curr = 1; curr < img->rx_size - 1; curr++) {
- switch (*(src_ptr + curr)) {
- case HDLC_ESCAPE:
- curr++;
- dest_len++;
- break;
- default:
- dest_len++;
- break;
- }
- }
- /* Allocate meomory for decoded message*/
- dest_ptr = kmalloc(dest_len, GFP_KERNEL);
- /* Memory allocate fail.*/
- if (dest_ptr == NULL)
- return -ENOMEM;
- memset(dest_ptr, 0, dest_len);
- curr = 0;
- dest_len = 0;
- /* Do de-escape.*/
- for (curr = 1; curr < img->rx_size - 1; curr++) {
- switch (*(src_ptr + curr)) {
- case HDLC_ESCAPE:
- curr++;
- *(dest_ptr + dest_len) =
- *(src_ptr + curr) ^ HDLC_ESCAPE_MASK;
- break;
- default:
- *(dest_ptr + dest_len) = *(src_ptr + curr);
- break;
- }
- dest_len = dest_len + 1;
- }
- /* CRC Check. */
- if (bcrc)
- crc = sprd_crc_calc(dest_ptr, dest_len);
- else
- crc = sprd_crc_calc_fdl((unsigned short *)dest_ptr, dest_len);
- if (crc != CRC_16_L_OK) {
- pr_err("CRC error : 0x%X", crc);
- kfree(dest_ptr);
- return -EPERM;
- }
- memcpy(img->decoded_rx_b, dest_ptr, dest_len - CRC_CHECK_SIZE);
- img->decoded_rx_size = dest_len - CRC_CHECK_SIZE ;
- kfree(dest_ptr);
- return 0;
- }
- static int if_spi_send_modem_bin_execute_cmd
- (u8 *spi_ptr, u32 spi_size, u16 spi_type,
- u16 spi_crc, struct sprd_image_buf *sprd_img)
- {
- int i, retval;
- u16 send_packet_size;
- u8 *send_packet_data;
- u16 d1_crc;
- u16 d2_crc = spi_crc;
- u16 type = spi_type;
- //dklee test add log
- static u16 data_sent_count = 0;
- static u16 err_count = 0;
- /* D1 */
- send_packet_size = spi_size; /* u32 -> u16 */
- sprd_img->tx_size = 6;
- M_16_SWAP(d2_crc);
- memcpy(sprd_img->tx_b, &send_packet_size, sizeof(send_packet_size));
- memcpy((sprd_img->tx_b+2), &type, sizeof(type));
- memcpy((sprd_img->tx_b+4), &d2_crc, sizeof(d2_crc));
- d1_crc = sprd_crc_calc_fdl
- ((unsigned short *)sprd_img->tx_b, sprd_img->tx_size);
- M_16_SWAP(d1_crc);
- memcpy((sprd_img->tx_b+6), &d1_crc, sizeof(d1_crc));
- sprd_img->tx_size += 2;
- if (down_timeout(&p_spild->srdy_sem, 2 * HZ)) {
- pr_err("(%d) SRDY TimeOUT!!! SRDY : %d, SEM : %d\n",
- __LINE__, gpio_get_value(p_spild->gpio_modem_bin_srdy),
- p_spild->srdy_sem.count);
- goto err;
- }
- #if 0//def USE_SPI_HALF_DUPLEX
- retval = spi_tx_rx_sync
- (sprd_img->tx_b, NULL , sprd_img->tx_size);
- #else
- retval = spi_tx_rx_sync
- (sprd_img->tx_b, sprd_img->rx_b, sprd_img->tx_size);
- #endif
- if (retval != 0) {
- pr_err("(%d) spi sync error : %d\n",
- __LINE__, retval);
- goto err;
- }
- if ((type == 0x0003) || (type == 0x0004)) {
- pr_err("D2 Skip!!\n");
- goto ACK;
- }
- /* D2 */
- send_packet_data = spi_ptr;
- if (down_timeout(&p_spild->srdy_sem, 2 * HZ)) {
- pr_err("(%d) SRDY TimeOUT!!! SRDY : %d, SEM : %d\n",
- __LINE__, gpio_get_value(p_spild->gpio_modem_bin_srdy),
- p_spild->srdy_sem.count);
- goto err;
- }
- data_sent_count++;
- #if 0//def USE_SPI_HALF_DUPLEX
- retval = spi_tx_rx_sync
- (send_packet_data, NULL , send_packet_size);
- #else
- retval = spi_tx_rx_sync
- (send_packet_data, sprd_img->rx_b, send_packet_size);
- #endif
- if (retval != 0) {
- pr_err("(%d) spi sync error : %d\n",
- __LINE__, retval);
- goto err;
- }
- ACK:
- if (p_spild->is_cp_reset) {
- while (!gpio_get_value(p_spild->gpio_modem_bin_srdy))
- ;
- } else {
- if (down_timeout(&p_spild->srdy_sem, 2 * HZ)) {
- pr_err("(%d) SRDY TimeOUT!!! SRDY : %d, SEM : %d\n",
- __LINE__, gpio_get_value(p_spild->gpio_modem_bin_srdy),
- p_spild->srdy_sem.count);
- pr_err("[SPI DUMP] TX_D1(%d) : [%02x %02x %02x %02x %02x %02x %02x %02x]\n",
- sprd_img->tx_size, sprd_img->tx_b[0], sprd_img->tx_b[1],
- sprd_img->tx_b[2], sprd_img->tx_b[3], sprd_img->tx_b[4],
- sprd_img->tx_b[5], sprd_img->tx_b[6],
- sprd_img->tx_b[7]);
- pr_err("[SPI DUMP] TX_D2(%d) : [%02x %02x %02x %02x %02x %02x %02x %02x]\n",
- send_packet_size, spi_ptr[0], spi_ptr[1], spi_ptr[2],
- spi_ptr[3], spi_ptr[4], spi_ptr[5], spi_ptr[6],
- spi_ptr[7]);
- /* WA (CP Reset) jongmoon.suh */
- if (gpio_get_value(p_spild->gpio_modem_bin_srdy))
- ;
- else
- goto err;
- }
- }
- #if 0//def USE_SPI_HALF_DUPLEX
- retval = spi_tx_rx_sync(NULL, sprd_img->rx_b, 8);
- #else
- memset(sprd_img->tx_b, 0, SPRD_BLOCK_SIZE+10);
- retval = spi_tx_rx_sync(sprd_img->tx_b, sprd_img->rx_b, 8);
- #endif
- if (retval != 0) {
- pr_err("(%d) spi sync error : %d\n",
- __LINE__, retval);
- goto err;
- }
- memcpy(sprd_img->decoded_rx_b, sprd_img->rx_b, 4);
- if ((*(sprd_img->decoded_rx_b+0) == 0x00) && \
- (*(sprd_img->decoded_rx_b+1) == 0x80) && \
- (*(sprd_img->decoded_rx_b+2) == 0x00) && \
- (*(sprd_img->decoded_rx_b+3) == 0x00)) {
- // pr_err("[SPRD] CP sent ACK");
- } else {
- pr_err("Transfer ACK error! srdy_sem = %d err count [%d]\n",
- p_spild->srdy_sem.count, ++err_count);
- pr_err("sent count = [%d]\n", data_sent_count);
- pr_err("[SPI DUMP] RX(%d) : [ ", sprd_img->rx_size);
- for (i = 0; i < 15; i++)
- pr_err("%02x ", *((u8 *)(sprd_img->rx_b + i)));
- pr_err("]");
- pr_err("[SPI DUMP] TX_D1(%d) : [%02x %02x %02x %02x %02x %02x %02x %02x]\n",
- sprd_img->tx_size, sprd_img->tx_b[0], sprd_img->tx_b[1],
- sprd_img->tx_b[2], sprd_img->tx_b[3], sprd_img->tx_b[4],
- sprd_img->tx_b[5], sprd_img->tx_b[6],
- sprd_img->tx_b[7]);
- pr_err("[SPI DUMP] TX_D2(%d) : [%02x %02x %02x %02x %02x %02x %02x %02x]\n",
- send_packet_size, spi_ptr[0], spi_ptr[1], spi_ptr[2],
- spi_ptr[3], spi_ptr[4], spi_ptr[5], spi_ptr[6],
- spi_ptr[7]);
- goto err;
- }
- return retval;
- err:
- return -EINVAL;
- }
- static int if_spi_send_modem_bin_xmit_img
- (enum image_type type, struct image_buf *img)
- {
- int retval = 0;
- struct sprd_image_buf sprd_img;
- unsigned int data_size;
- unsigned int send_size = 0;
- unsigned int rest_size = 0;
- unsigned int spi_size = 0;
- unsigned int address = 0;
- unsigned int fdl1_size = 0;
- /* No Translate */
- u16 crc = 0;
- u16 spi_type = 0;
- unsigned char *spi_ptr;
- unsigned char *ptr;
- int i, j = 0;
- u16 sprd_packet_size = SPRD_BLOCK_SIZE;
- sprd_img.tx_b = kmalloc(SPRD_BLOCK_SIZE*2, GFP_KERNEL);
- if (!sprd_img.tx_b) {
- pr_err("(%d) tx_b kmalloc fail.",
- __LINE__);
- return -ENOMEM;
- }
- memset(sprd_img.tx_b, 0, SPRD_BLOCK_SIZE*2);
- sprd_img.rx_b = kmalloc(SPRD_BLOCK_SIZE*2, GFP_KERNEL);
- if (!sprd_img.rx_b) {
- pr_err("(%d) rx_b kmalloc fail.",
- __LINE__);
- retval = -ENOMEM;
- goto err3;
- }
- memset(sprd_img.rx_b, 0, SPRD_BLOCK_SIZE*2);
- sprd_img.encoded_tx_b = kmalloc(SPRD_BLOCK_SIZE*2, GFP_KERNEL);
- if (!sprd_img.encoded_tx_b) {
- pr_err("(%d) encoded_tx_b kmalloc fail.",
- __LINE__);
- retval = -ENOMEM;
- goto err2;
- }
- memset(sprd_img.encoded_tx_b, 0, SPRD_BLOCK_SIZE*2);
- sprd_img.decoded_rx_b = kmalloc(SPRD_BLOCK_SIZE*2, GFP_KERNEL);
- if (!sprd_img.decoded_rx_b) {
- pr_err("(%d) encoded_rx_b kmalloc fail.",
- __LINE__);
- retval = -ENOMEM;
- goto err1;
- }
- memset(sprd_img.decoded_rx_b, 0, SPRD_BLOCK_SIZE*2);
- pr_debug("(%d) if_spi_send_modem_bin_xmit_img type : %d.\n",
- __LINE__, type);
- memcpy(&fdl1_size, (void *)(p_spild->p_virtual_buff + 4), 4);
- switch (type) {
- case MODEM_MAIN:
- memcpy(&img->address, (void *)(p_spild->p_virtual_buff + 8), 4);
- memcpy(&img->length ,
- (void *)(p_spild->p_virtual_buff + 12), 4);
- img->buf = (unsigned char *)
- (p_spild->p_virtual_buff + 0x30 + fdl1_size);
- img->offset = img->length + fdl1_size + 0x30;
- pr_debug(
- "(%d) if_spi_send_modem_bin_xmit_img save MAIN to img.\n",
- __LINE__);
- break;
- case MODEM_DSP:
- memcpy(&img->address,
- (void *)(p_spild->p_virtual_buff + 16), 4);
- memcpy(&img->length,
- (void *)(p_spild->p_virtual_buff + 20), 4);
- img->buf = (unsigned char *)
- (p_spild->p_virtual_buff + img->offset);
- img->offset += img->length;
- pr_debug("(%d) if_spi_send_modem_bin_xmit_img save DSP to img.\n",
- __LINE__);
- break;
- case MODEM_NV:
- memcpy(&img->address,
- (void *)(p_spild->p_virtual_buff + 24), 4);
- memcpy(&img->length,
- (void *)(p_spild->p_virtual_buff + 28), 4);
- img->buf = (unsigned char *)
- (p_spild->p_virtual_buff + img->offset);
- img->offset += img->length;
- pr_debug("(%d) if_spi_send_modem_bin_xmit_img save NV to img.\n",
- __LINE__);
- break;
- case MODEM_EFS:
- memcpy(&img->address,
- (void *)(p_spild->p_virtual_buff + 32), 4);
- memcpy(&img->length,
- (void *)(p_spild->p_virtual_buff + 36), 4);
- img->buf = (unsigned char *)
- (p_spild->p_virtual_buff + img->offset);
- img->offset += img->length;
- pr_debug("(%d) if_spi_send_modem_bin_xmit_img save EFS to img.\n",
- __LINE__);
- break;
- case MODEM_RUN:
- memset(sprd_img.encoded_tx_b, 0, SPRD_BLOCK_SIZE*2);
- sprd_img.encoded_tx_size = 0;
- spi_type = 0x0004;
- crc = 0;
- spi_ptr = sprd_img.encoded_tx_b;
- spi_size = sprd_img.encoded_tx_size;
- retval = if_spi_send_modem_bin_execute_cmd
- (spi_ptr, spi_size, spi_type, crc, &sprd_img);
- if (retval < 0) {
- pr_err("(%d) if_spi_send_modem_bin_execute_cmd fail : %d",
- __LINE__, retval);
- goto err0;
- }
- return retval;
- default:
- pr_err("(%d) if_spi_send_modem_bin_xmit_img wrong : %d.",
- __LINE__, type);
- goto err0;
- }
- pr_debug("(%d) Start send img. size : %d\n",
- __LINE__, img->length);
- ptr = img->buf;
- data_size = sprd_packet_size;
- rest_size = img->length;
- address = img->address;
- M_32_SWAP(img->address);
- M_32_SWAP(img->length);
- /* Send Transfer Start */
- sprd_img.tx_size = 8;
- memcpy((sprd_img.tx_b+0), &img->address, sizeof(img->address));
- memcpy((sprd_img.tx_b+4), &img->length, sizeof(img->length));
- spi_type = 0x0001;
- crc = sprd_crc_calc_fdl
- ((unsigned short *)sprd_img.tx_b, sprd_img.tx_size);
- memcpy(sprd_img.encoded_tx_b, sprd_img.tx_b, sprd_img.tx_size);
- sprd_img.encoded_tx_size = sprd_img.tx_size;
- spi_ptr = sprd_img.encoded_tx_b;
- spi_size = sprd_img.encoded_tx_size;
- pr_debug("(%d) [Transfer Start, Type = %d, Packet = %d]\n",
- __LINE__, type, sprd_packet_size);
- retval = if_spi_send_modem_bin_execute_cmd
- (spi_ptr, spi_size, spi_type, crc, &sprd_img);
- if (retval < 0) {
- pr_err("(%d) if_spi_send_modem_bin_execute_cmd fail : %d",
- __LINE__, retval);
- goto err0;
- }
- M_32_SWAP(img->length);
- /* Send Data */
- for (i = 0; send_size < img->length; i++) {
- if (rest_size < sprd_packet_size)
- data_size = rest_size;
- sprd_img.encoded_tx_size = sprd_packet_size;
- for (j = 0; j < data_size; j++)
- *(sprd_img.encoded_tx_b+j) = *(ptr + j);
- spi_type = 0x0002;
- crc = sprd_crc_calc_fdl
- ((unsigned short *)sprd_img.encoded_tx_b,
- sprd_img.encoded_tx_size);
- spi_ptr = sprd_img.encoded_tx_b;
- spi_size = sprd_img.encoded_tx_size;
- retval = if_spi_send_modem_bin_execute_cmd
- (spi_ptr, spi_size, spi_type, crc, &sprd_img);
- if (retval < 0) {
- pr_err("(%d) if_spi_send_modem_bin_execute_cmd fail : %d, %d",
- __LINE__, retval, i);
- goto err0;
- }
- send_size += data_size;
- rest_size -= data_size;
- ptr += data_size;
- if (!(i % 100))
- pr_debug("(%d) [%d] 0x%x size done, rest size: 0x%x\n",
- __LINE__, i, send_size, rest_size);
- }
- /* Send Transfer End */
- memset(sprd_img.encoded_tx_b, 0, SPRD_BLOCK_SIZE * 2);
- sprd_img.encoded_tx_size = 0;
- spi_type = 0x0003;
- crc = 0;
- spi_ptr = sprd_img.encoded_tx_b;
- spi_size = sprd_img.encoded_tx_size;
- pr_debug("(%d) [Transfer END]\n", __LINE__);
- retval = if_spi_send_modem_bin_execute_cmd
- (spi_ptr, spi_size, spi_type, crc, &sprd_img);
- if (retval < 0) {
- pr_err("(%d) if_spi_send_modem_bin_execute_cmd fail : %d",
- __LINE__, retval);
- goto err0;
- }
- err0:
- kfree(sprd_img.decoded_rx_b);
- err1:
- kfree(sprd_img.encoded_tx_b);
- err2:
- kfree(sprd_img.rx_b);
- err3:
- kfree(sprd_img.tx_b);
- return retval;
- }
- static void if_spi_send_modem_bin(struct work_struct *send_modem_w)
- {
- int retval;
- struct image_buf img;
- unsigned long tick1, tick2 = 0;
- tick1 = jiffies_to_msecs(jiffies);
- retval = if_spi_send_modem_bin_xmit_img(MODEM_MAIN, &img);
- if (retval < 0) {
- pr_err("(%d) if_spi_send_modem_bin_xmit_img fail : %d",
- __LINE__, retval);
- goto err;
- }
- retval = if_spi_send_modem_bin_xmit_img(MODEM_DSP, &img);
- if (retval < 0) {
- pr_err("(%d) if_spi_send_modem_bin_xmit_img fail : %d",
- __LINE__, retval);
- goto err;
- }
- retval = if_spi_send_modem_bin_xmit_img(MODEM_NV, &img);
- if (retval < 0) {
- pr_err("(%d) if_spi_send_modem_bin_xmit_img fail : %d",
- __LINE__, retval);
- goto err;
- }
- retval = if_spi_send_modem_bin_xmit_img(MODEM_EFS, &img);
- if (retval < 0) {
- pr_err("(%d) if_spi_send_modem_bin_xmit_img fail : %d",
- __LINE__, retval);
- goto err;
- }
- retval = if_spi_send_modem_bin_xmit_img(MODEM_RUN, &img);
- if (retval < 0) {
- pr_err("(%d) if_spi_send_modem_bin_xmit_img fail : %d",
- __LINE__, retval);
- goto err;
- }
- p_spild->send_modem_spi = 0;
- p_spild->is_cp_reset = 0;
- tick2 = jiffies_to_msecs(jiffies);
- pr_info("Downloading takes %lu msec\n", (tick2-tick1));
- spi_init_ipc(p_spild);
- sprd_boot_done = 1;
- p_spild->ril_send_cnt = 0;
- p_spild->spi_state = SPI_STATE_IDLE;
- err:
- return;
- }
- static inline int _request_mem(struct spi_v_buff *od,
- struct platform_device *pdev)
- {
- if (!p_spild->p_virtual_buff) {
- od->mmio = vmalloc(od->size);
- if (!od->mmio) {
- pr_err("(%d) Failed to vmalloc size : %lu\n",
- __LINE__, od->size);
- return -EBUSY;
- } else {
- pr_err("(%d) vmalloc Done. mmio : 0x%08x\n",
- __LINE__, (u32)od->mmio);
- }
- }
- memset((void *)od->mmio, 0, od->size);
- p_spild->p_virtual_buff = od->mmio;
- return 0;
- }
- void spi_tx_timer_callback(unsigned long param)
- {
- if (p_spild->spi_state == SPI_STATE_TX_WAIT) {
- p_spild->spi_timer_tx_state = SPI_STATE_TIME_OVER;
- pr_err("[SPI] spi_tx_timer_callback -timer expires\n");
- }
- }
- void spi_rx_timer_callback(unsigned long param)
- {
- if (p_spild->spi_state == SPI_STATE_RX_WAIT) {
- p_spild->spi_timer_rx_state = SPI_STATE_TIME_OVER;
- pr_err("[SPI] spi_rx_timer_callback -timer expires\n");
- }
- }
- int spi_sema_init(void)
- {
- pr_info("[SPI] Srdy sema init\n");
- sema_init(&p_spild->srdy_sem, 0);
- p_spild->send_modem_spi = 1;
- return 0;
- }
- void spi_work(struct work_struct *work)
- {
- int signal_code;
- struct spi_work_type *spi_wq =
- container_of(work, struct spi_work_type, work);
- signal_code = spi_wq->signal_code;
- if (p_spild->spi_state == SPI_STATE_END
- || p_spild->spi_state == SPI_STATE_START) {
- kfree(spi_wq);
- return;
- }
- switch (signal_code) {
- case SPI_WORK_SEND:
- if (p_spild->spi_state == SPI_STATE_IDLE)
- {
- /* active the spi wakelock to prevent ap
- becoming deep sleep during waiting SUB_SRDY signal. Otherwise,
- may cause CP side TX queue full, because the MRDY keeping on
- HIGH_LEVEL*/
- if (!wake_lock_active(&p_spild->spi_wake_lock)){
- wake_lock(&p_spild->spi_wake_lock);
- pr_debug("[SPI] [%s](%d) spi_wakelock locked .\n",
- __func__, __LINE__);
- }
- spi_tx_work();
- }
- break;
- case SPI_WORK_RECEIVE:
- if (p_spild->spi_state == SPI_STATE_IDLE
- || p_spild->spi_state == SPI_STATE_TX_TERMINATE
- || p_spild->spi_state == SPI_STATE_RX_TERMINATE)
- spi_rx_work();
- break;
- default:
- pr_err("[SPI] ERROR(No signal_code in spi_work[%d])\n",
- signal_code);
- break;
- }
- kfree(spi_wq);
- if (wake_lock_active(&p_spild->spi_wake_lock)) {
- wake_unlock(&p_spild->spi_wake_lock);
- pr_debug("[SPI] [%s](%d) spi_wakelock unlocked .\n",
- __func__, __LINE__);
- }
- }
- static void get_mif_spi_dt_data(struct device *dev)
- {
- /* Initialize SPI pin value */
- p_spild->gpio_ipc_mrdy = of_get_named_gpio(dev->of_node, "mif_spi,gpio_ipc_mrdy", 0);
- if (gpio_is_valid(p_spild->gpio_ipc_mrdy)) {
- int ret;
- ret = gpio_request(p_spild->gpio_ipc_mrdy, "ipc_mrdy");
- if (ret)
- mif_err("Unable to request ipc_mrdy [%d]\n", p_spild->gpio_ipc_mrdy);
- else {
- gpio_tlmm_config(GPIO_CFG(p_spild->gpio_ipc_mrdy, GPIOMUX_FUNC_GPIO,
- GPIO_CFG_OUTPUT, GPIO_CFG_NO_PULL, GPIO_CFG_2MA),
- GPIO_CFG_ENABLE);
- gpio_direction_output(p_spild->gpio_ipc_mrdy, 0);
- }
- }
- else {
- mif_err("Failed to get is valid ipc_mrdy\n");
- }
- p_spild->gpio_ipc_srdy = of_get_named_gpio(dev->of_node, "mif_spi,gpio_ipc_srdy", 0);
- if (gpio_is_valid(p_spild->gpio_ipc_srdy)) {
- int ret;
- ret = gpio_request(p_spild->gpio_ipc_srdy, "ipc_srdy");
- if (ret)
- mif_err("Unable to request ipc_srdy [%d]\n", p_spild->gpio_ipc_srdy);
- else {
- gpio_tlmm_config(GPIO_CFG(p_spild->gpio_ipc_srdy, GPIOMUX_FUNC_GPIO,
- GPIO_CFG_INPUT, GPIO_CFG_PULL_DOWN, GPIO_CFG_2MA), GPIO_CFG_ENABLE);
- }
- }
- else {
- mif_err("Failed to get is valid ipc_srdy\n");
- }
- p_spild->gpio_ipc_sub_mrdy = of_get_named_gpio(dev->of_node, "mif_spi,gpio_ipc_sub_mrdy", 0);
- if (gpio_is_valid(p_spild->gpio_ipc_sub_mrdy)) {
- int ret;
- ret = gpio_request(p_spild->gpio_ipc_sub_mrdy, "ipc_sub_mrdy");
- if (ret)
- mif_err("Unable to request ipc_sub_mrdy [%d]\n", p_spild->gpio_ipc_sub_mrdy);
- else {
- gpio_tlmm_config(GPIO_CFG(p_spild->gpio_ipc_sub_mrdy, GPIOMUX_FUNC_GPIO,
- GPIO_CFG_OUTPUT, GPIO_CFG_NO_PULL, GPIO_CFG_2MA),GPIO_CFG_ENABLE);
- gpio_direction_output(p_spild->gpio_ipc_sub_mrdy, 0);
- }
- }
- else {
- mif_err("Failed to get is valid ipc_sub_mrdy\n");
- }
- p_spild->gpio_ipc_sub_srdy = of_get_named_gpio(dev->of_node, "mif_spi,gpio_ipc_sub_srdy", 0);
- if (gpio_is_valid(p_spild->gpio_ipc_sub_srdy)) {
- int ret;
- ret = gpio_request(p_spild->gpio_ipc_sub_srdy, "ipc_sub_srdy");
- if (ret)
- mif_err("Unable to request ipc_sub_srdy [%d]\n", p_spild->gpio_ipc_sub_srdy);
- else {
- gpio_tlmm_config(GPIO_CFG(p_spild->gpio_ipc_sub_srdy, GPIOMUX_FUNC_GPIO,
- GPIO_CFG_INPUT, GPIO_CFG_PULL_DOWN, GPIO_CFG_2MA),GPIO_CFG_ENABLE);
- }
- }
- else {
- mif_err("Failed to get is valid ipc_sub_srdy\n");
- }
- p_spild->gpio_cp_dump_int = of_get_named_gpio(dev->of_node, "mif_spi,gpio_cp_dump_int", 0);
- if (gpio_is_valid(p_spild->gpio_cp_dump_int)) {
- int ret;
- ret = gpio_request(p_spild->gpio_cp_dump_int, "cp_dump_int");
- if (ret)
- mif_err("Unable to request cp_dump_int [%d]\n", p_spild->gpio_cp_dump_int);
- else {
- gpio_tlmm_config(GPIO_CFG(p_spild->gpio_cp_dump_int, GPIOMUX_FUNC_GPIO,
- GPIO_CFG_INPUT, GPIO_CFG_PULL_DOWN, GPIO_CFG_2MA), GPIO_CFG_ENABLE);
- }
- }
- else {
- mif_err("Failed to get is valid cp_dump_int\n");
- }
- p_spild->gpio_modem_bin_srdy = p_spild->gpio_ipc_srdy;
- mif_info("gpio_mrdy:%d, gpio_srdy:%d, gpio_submrdy:%d, gpio_subsrdy:%d, gpio_cpdump_int:%d\n",
- p_spild->gpio_ipc_mrdy, p_spild->gpio_ipc_srdy,
- p_spild->gpio_ipc_sub_mrdy, p_spild->gpio_ipc_sub_srdy,
- p_spild->gpio_cp_dump_int);
- }
- static int __devinit if_spi_platform_probe(struct platform_device *pdev)
- {
- int ret;
- struct spi_v_buff *od;
- struct spi_platform_data *pdata = NULL;
- // struct task_struct *th;
- struct link_device *ld = &p_spild->ld;
- pr_debug("[%s]\n", __func__);
- pdata = pdev->dev.platform_data;
- if (!pdata) {
- pr_err("No platform data\n");
- ret = -EINVAL;
- goto err;
- }
- od = kzalloc(sizeof(struct spi_v_buff), GFP_KERNEL);
- if (!od) {
- pr_err("(%d) failed to allocate device\n",
- __LINE__);
- ret = -ENOMEM;
- goto err;
- }
- od->base = 0;
- od->size = SZ_16M; /* 16M */
- if (p_spild->p_virtual_buff)
- od->mmio = p_spild->p_virtual_buff;
- ret = _request_mem(od, pdev);
- if (ret)
- {
- kfree(od);
- goto err;
- }
- sema_init(&p_spild->srdy_sem, 0);
- INIT_WORK(&p_spild->send_modem_w,
- if_spi_send_modem_bin);
- platform_set_drvdata(pdev, od);
- wake_lock_init(&p_spild->spi_wake_lock,
- WAKE_LOCK_SUSPEND,
- "samsung-spiwakelock");
- /* Register CP_DUMP_INT interrupt handler */
- ret = spi_register_isr(gpio_to_irq(p_spild->gpio_cp_dump_int),
- spi_cp_dump_irq_handler,
- IRQF_TRIGGER_RISING,
- "spi_cp_dump_rising",
- ld);
- if (ret)
- goto err;
- /* Register SPI Srdy interrupt handler */
- ret = spi_register_isr(gpio_to_irq(p_spild->gpio_ipc_srdy),
- spi_srdy_irq_handler,
- IRQF_TRIGGER_RISING,
- "spi_srdy_rising",
- ld);
- if (ret)
- goto err;
- /* Register SPI SubSrdy interrupt handler */
- ret = spi_register_isr(gpio_to_irq(p_spild->gpio_ipc_sub_srdy),
- spi_subsrdy_irq_handler,
- IRQF_TRIGGER_RISING,
- "spi_subsrdy_rising",
- ld);
- if (ret)
- goto err;
- p_spild->boot_done = 1;
- pr_info("[%s] Done\n", __func__);
- err:
- /* _release(od); */
- return 0;
- }
- static int __devexit if_spi_platform_remove(struct platform_device *pdev)
- {
- struct spi_v_buff *od = platform_get_drvdata(pdev);
- platform_set_drvdata(pdev, NULL);
- kfree(od);
- return 0;
- }
- static int if_spi_probe(struct spi_device *spi)
- {
- int ret=0;
- p_spi = spi;
- mif_info("if-spi->mode : [%d]\n", p_spi->mode);
- mif_info("if_spi->bits_per_word : [%d]\n", p_spi->bits_per_word);
- #if 0
- p_spi->mode = SPI_MODE_1;
- p_spi->bits_per_word = 32;
- ret = spi_setup(p_spi);
- if (ret != 0) {
- pr_err("[%s] spi_setup ERROR : %d\n", __func__, ret);
- return ret;
- }
- #endif
- get_mif_spi_dt_data(&spi->dev);
- pr_info("[%s]\n", __func__);
- return ret;
- }
- static int if_spi_remove(struct spi_device *spi)
- {
- return 0;
- }
- static struct platform_driver if_spi_platform_driver = {
- .probe = if_spi_platform_probe,
- .remove = __devexit_p(if_spi_platform_remove),
- .driver = {
- .name = "if_spi_platform_driver",
- },
- };
- static const struct of_device_id if_spi_match_table[] = {
- { .compatible = "if_spi_comp",
- },
- {}
- };
- static struct spi_driver if_spi_driver = {
- .probe = if_spi_probe,
- .remove = __devexit_p(if_spi_remove),
- .driver = {
- .name = "if_spi_driver",
- .of_match_table = if_spi_match_table,
- .owner = THIS_MODULE,
- },
- };
- static int spi_link_init(void)
- {
- int ret;
- ret = spi_register_driver(&if_spi_driver);
- if (ret < 0) {
- pr_err("spi_register_driver() fail : %d\n", ret);
- goto err;
- }
- ret = platform_driver_register(&if_spi_platform_driver);
- if (ret < 0) {
- pr_err("[%s] platform_driver_register ERROR : %d\n",
- __func__, ret);
- goto err;
- }
- /* creat work queue thread */
- p_spild->spi_modem_wq = create_singlethread_workqueue("spi_modem_wq");
- if (!p_spild->spi_modem_wq) {
- pr_err("[%s] get workqueue thread fail\n",
- __func__);
- ret = -ENOMEM;
- goto err;
- }
- pr_info("[%s] Done\n", __func__);
- err:
- return ret;
- }
- /*=====================================
- * Description spi restart for CP slient reset
- =====================================*/
- void if_spi_thread_restart(void)
- {
- p_spild->send_modem_spi = 1;
- p_spild->is_cp_reset = 1;
- sprd_boot_done = 0;
- pr_info("[IF_SPI] if_spi_thread_restart\n");
- return;
- }
- EXPORT_SYMBOL(if_spi_thread_restart);
- struct link_device *spi_create_link_device(struct platform_device *pdev)
- {
- struct spi_link_device *spild = NULL;
- struct link_device *ld;
- struct modem_data *pdata;
- int ret;
- int i;
- /* Get the platform data */
- pdata = (struct modem_data *)pdev->dev.platform_data;
- if (!pdata) {
- pr_err("[LNK/E] <%s> pdata == NULL\n", __func__);
- goto err2;
- }
- mif_err("[LNK] <%s> link device = %s\n", __func__, pdata->link_name);
- mif_err("[LNK] <%s> modem = %s\n", __func__, pdata->name);
- /* Alloc SPI link device structure */
- p_spild = spild = kzalloc(sizeof(struct spi_link_device), GFP_KERNEL);
- if (!spild) {
- pr_err("[LNK/E] <%s> Failed to kzalloc()\n", __func__);
- goto err2;
- }
- ld = &spild->ld;
- /* Extract modem data and SPI control data from the platform data */
- ld->name = "spi";
- if (ld->aligned)
- pr_err("[LNK] <%s> Aligned access is required!!!\n", __func__);
- /* Set attributes as a link device */
- ld->send = spi_send;
- ld->ioctl = spi_ioctl; //SPI_SETUP
- INIT_LIST_HEAD(&ld->list);
- skb_queue_head_init(&ld->sk_fmt_tx_q);
- skb_queue_head_init(&ld->sk_raw_tx_q);
- skb_queue_head_init(&ld->sk_rfs_tx_q);
- spild->skb_txq[IPC_FMT] = &ld->sk_fmt_tx_q;
- spild->skb_txq[IPC_RAW] = &ld->sk_raw_tx_q;
- spild->skb_txq[IPC_RFS] = &ld->sk_rfs_tx_q;
- spild->spi_wq = create_singlethread_workqueue("spi_wq");
- if (!spild->spi_wq) {
- pr_err("[LNK/E] <%s> Fail to create workqueue for spi_wq\n",
- __func__);
- goto err1;
- }
- spild->spi_state = SPI_STATE_END;
- spild->max_ipc_dev = IPC_RFS+1; /* FMT, RAW, RFS */
- for (i = 0; i < spild->max_ipc_dev; i++)
- skb_queue_head_init(&spild->skb_rxq[i]);
- /* Prepare a clean buffer for SPI access */
- spild->buff = kzalloc(SPI_MAX_PACKET_SIZE, GFP_KERNEL);
- spild->sync_buff = kzalloc(SPI_MAX_PACKET_SIZE, GFP_KERNEL);
- memset(spild->buff , 0, SPI_MAX_PACKET_SIZE);
- memset(spild->sync_buff , 0, SPI_MAX_PACKET_SIZE);
- if (!spild->buff) {
- pr_err("[LNK/E] <%s> Failed to alloc spild->buff\n", __func__);
- goto err;
- }
- /* Create SPI Timer */
- init_timer(&spild->spi_tx_timer);
- spild->spi_tx_timer.expires = jiffies +
- ((SPI_TIMER_TX_WAIT_TIME * HZ) / 1000);
- spild->spi_tx_timer.data = 0;
- spild->spi_tx_timer.function = spi_tx_timer_callback;
- init_timer(&spild->spi_rx_timer);
- spild->spi_rx_timer.expires = jiffies +
- ((SPI_TIMER_RX_WAIT_TIME * HZ) / 1000);
- spild->spi_rx_timer.data = 0;
- spild->spi_rx_timer.function = spi_rx_timer_callback;
- /* Create SPI device */
- ret = spi_link_init();
- if (ret)
- goto err;
- return ld;
- err:
- kfree(spild->buff);
- err1:
- kfree(spild);
- err2:
- return NULL;
- }
|