123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085 |
- /*
- * This file is part of FH8620 BSP for RT-Thread distribution.
- *
- * Copyright (c) 2016 Shanghai Fullhan Microelectronics Co., Ltd.
- * All rights reserved
- *
- * 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.,
- * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
- *
- * Visit http://www.fullhan.com to get contact with Fullhan.
- *
- * Change Logs:
- * Date Author Notes
- */
- #include "string.h"
- #include "inc/fh_sdio.h"
- #include "fh_arch.h"
- //#include "interrupt.h"
- #define SDIO_PRINTF rt_kprintf
- #define SDC_USE_IDMA
- #define INSTRUCTIONS_PER_USEC 1000
- #define CMD_TIMEOUT_USEC 100000
- #define DATA_READY_TIMEOUT_USEC 200000
- #define DMA_TRANSFER_TIMEOUT_TICKS 300
- #define DATA_TRANSFER_OVER_TIMEOUT_USEC 1000
- #define ACMD41_RETRY_COUNT 1000//100000
- #define CIU_CLK 50000//25000//25000 //27MHz
- #define MMC_FOD_VALUE 125 /* 125 KHz */
- #define NORM_FOD_VALUE 25000//5000//25000 /* 25 MHz */
- #define MMC_FOD_DIVIDER_VALUE (((CIU_CLK+MMC_FOD_VALUE*2-1)/(MMC_FOD_VALUE*2)))
- #ifdef SDCARD_CLK_DIVIDER
- #define ONE_BIT_BUS_FREQ SDCARD_CLK_DIVIDER
- #else
- #define ONE_BIT_BUS_FREQ (((CIU_CLK)/(NORM_FOD_VALUE*2)))
- #endif
- static unsigned int sdc_clk_divider = ONE_BIT_BUS_FREQ;
- static sdc_t sdc_array[2];
- static void plat_loop(unsigned int macrosecond)
- {
- unsigned int clk;
-
- while (macrosecond-- > 0) {
- for(clk=INSTRUCTIONS_PER_USEC; clk>0; clk--);
- }
- }
- static int synopmob_execute_command(unsigned int base, unsigned int cmd_register, unsigned int arg_register)
- {
- unsigned int retries = CMD_TIMEOUT_USEC;
- synopmob_set_register(base + RINTSTS, 0xfffe); //clear interrupts, FIXME
- synopmob_set_register(base+CMDARG, arg_register);
- synopmob_set_register(base+CMD, cmd_register | (0x80000000|0x20000000/*fixed to use hold*/));
- while (retries-- > 0) {
- if (!(synopmob_read_register(base+CMD) & 0x80000000/*CMD done bit*/))
- return 0;
- plat_loop(1);
- }
-
- return ERRCMDRETRIESOVER;
- }
- static int synopmob_wait_command_done(unsigned int base, unsigned int* inst, unsigned int flag)
- {
- unsigned int retries = CMD_TIMEOUT_USEC;
- unsigned int sts;
- while (retries-- > 0) {
- sts = synopmob_read_register(base+RINTSTS);
- if (sts && ((sts & flag) == flag) ) {
- *inst = sts;
- return 0;
- }
- plat_loop(1);
- }
- return ERRCMDRETRIESOVER;
- }
- static int synopmob_wait_data_ready(unsigned int base)
- {
- unsigned int retries = DATA_READY_TIMEOUT_USEC;
- while (retries-- > 0) {
- if (!((synopmob_read_register(base+STATUS)) & 0x00000200)) {
- return 0;
- }
-
- plat_loop(1);
- }
- return ERRDATANOTREADY;
- }
- static int synopmob_handle_standard_rinsts(unsigned int raw_int_stat)
- {
- int error_status = 0;
-
- if ( raw_int_stat & INTMASK_ERROR) {
- if (raw_int_stat & INTMSK_RESP_ERR) {
- error_status = ERRRESPRECEP;
- }
- if (raw_int_stat & INTMSK_RCRC) {
- error_status = ERRRESPCRC;
- }
- if (raw_int_stat & INTMSK_DCRC) {
- error_status = ERRDCRC;
- }
- if (raw_int_stat & INTMSK_RTO) {
- error_status = ERRRESPTIMEOUT;
- }
- if (raw_int_stat & INTMSK_DTO) {
- error_status = ERRDRTIMEOUT;
- }
- if (raw_int_stat & INTMSK_HTO) {
- error_status = ERRUNDERWRITE;
- }
- if (raw_int_stat & INTMSK_FRUN) {
- error_status = ERROVERREAD;
- }
- if (raw_int_stat & INTMSK_HLE) {
- error_status = ERRHLE;
- }
- if (raw_int_stat & INTMSK_SBE) {
- error_status = ERRSTARTBIT;
- }
- if (raw_int_stat & INTMSK_EBE) {
- error_status = ERRENDBITERR;
- }
- }
- //SDIO_PRINTF("------- %s, line %d raw_int_stat = %08x-------\n", __FUNCTION__, __LINE__, raw_int_stat);
- return error_status;
- }
- static int synopmob_check_r1_resp(unsigned int the_response)
- {
- int retval = 0;
-
- if (the_response & R1CS_ERROR_OCCURED_MAP) {
- if (the_response & R1CS_ADDRESS_OUT_OF_RANGE) {
- retval = ERRADDRESSRANGE;
- } else if (the_response & R1CS_ADDRESS_MISALIGN) {
- retval = ERRADDRESSMISALIGN;
- } else if (the_response & R1CS_BLOCK_LEN_ERR) {
- retval = ERRBLOCKLEN;
- } else if (the_response & R1CS_ERASE_SEQ_ERR) {
- retval = ERRERASESEQERR;
- } else if (the_response & R1CS_ERASE_PARAM) {
- retval = ERRERASEPARAM;
- } else if (the_response & R1CS_WP_VIOLATION) {
- retval = ERRPROT;
- } else if (the_response & R1CS_CARD_IS_LOCKED) {
- retval = ERRCARDLOCKED;
- } else if (the_response & R1CS_LCK_UNLCK_FAILED) {
- retval = ERRCARDLOCKED;
- } else if (the_response & R1CS_COM_CRC_ERROR) {
- retval = ERRCRC;
- } else if (the_response & R1CS_ILLEGAL_COMMAND) {
- retval = ERRILLEGALCOMMAND;
- } else if (the_response & R1CS_CARD_ECC_FAILED) {
- retval = ERRECCFAILED;
- } else if (the_response & R1CS_CC_ERROR) {
- retval = ERRCCERR;
- } else if (the_response & R1CS_ERROR) {
- retval = ERRUNKNOWN;
- } else if (the_response & R1CS_UNDERRUN) {
- retval = ERRUNDERRUN;
- } else if (the_response & R1CS_OVERRUN) {
- retval = ERROVERRUN;
- } else if (the_response & R1CS_CSD_OVERWRITE) {
- retval = ERRCSDOVERWRITE;
- } else if (the_response & R1CS_WP_ERASE_SKIP) {
- retval = ERRPROT;
- } else if (the_response & R1CS_ERASE_RESET) {
- retval = ERRERASERESET;
- } else if (the_response & R1CS_SWITCH_ERROR) {
- retval = ERRFSMSTATE;
- }
- }
-
- return retval;
- }
- static int synopmob_check_r5_resp(unsigned int the_resp)
- {
- int ret = 0;
-
- if (the_resp & R5_IO_ERR_BITS) {
- if (the_resp & R5_IO_CRC_ERR) {
- ret = ERRDCRC;
- } else if (the_resp & R5_IO_BAD_CMD) {
- ret = ERRILLEGALCOMMAND;
- } else if (the_resp & R5_IO_GEN_ERR) {
- ret = ERRUNKNOWN;
- } else if (the_resp & R5_IO_FUNC_ERR) {
- ret = ERRBADFUNC;
- } else if (the_resp & R5_IO_OUT_RANGE) {
- ret = ERRADDRESSRANGE;
- }
- }
-
- return ret;
- }
- static int sd_send_cmd0(sdc_t* sdc)
- {
- int ret;
- unsigned int intst;
- unsigned int base = sdc->ip_base;
- ret = synopmob_execute_command(base, 0x4000, 0);
- if (!ret) {
- ret = synopmob_wait_command_done(base, &intst, INTMSK_CMD_DONE);
- if (!ret) {
- synopmob_set_register(base+RINTSTS, intst);
- return synopmob_handle_standard_rinsts(intst);
- }
- }
- return ret;
- }
- static int sd_send_cmd2(sdc_t* sdc)
- {
- int ret;
- unsigned int intst;
- unsigned int base = sdc->ip_base;
- ret = synopmob_execute_command(base, 0xC2, 0);
- if (!ret) {
- ret = synopmob_wait_command_done(base, &intst, INTMSK_CMD_DONE);
- if (!ret) {
- synopmob_set_register(base+RINTSTS, intst);
- return synopmob_handle_standard_rinsts(intst);
- }
- }
- return ret;
- }
- static int sd_send_cmd3(sdc_t* sdc)
- {
- int ret;
- unsigned int intst;
- unsigned int resp;
- unsigned int base = sdc->ip_base;
- ret = synopmob_execute_command(base, 0x43, 0);
- if (!ret) {
- ret = synopmob_wait_command_done(base, &intst, INTMSK_CMD_DONE);
- if (!ret) {
- synopmob_set_register(base+RINTSTS, intst);
- ret = synopmob_handle_standard_rinsts(intst);
- if (!ret) {
- resp = synopmob_read_register(base+RESP0);
- sdc->rca = resp >> 16;
- resp = (resp & 0x1fff) | (((resp>>13)&1)<<19) | (((resp>>14)&3)<<22);
- return synopmob_check_r1_resp(resp);
- }
- }
- }
- return ret;
- }
- static int sd_send_cmd_r1(sdc_t* sdc, unsigned int cmd, unsigned int arg, unsigned int buzy)
- {
- int ret;
- unsigned int intst;
- unsigned int resp;
- unsigned int base = sdc->ip_base;
-
- ret = synopmob_execute_command(base, cmd, arg);
- if (!ret) {
- ret = synopmob_wait_command_done(base, &intst, INTMSK_CMD_DONE);
- if (!ret) {
- synopmob_set_register(base+RINTSTS, intst);
- ret = synopmob_handle_standard_rinsts(intst);
- if (!ret) {
- resp = synopmob_read_register(base+RESP0);
- ret = synopmob_check_r1_resp(resp);
- if (buzy && !ret) {
- ret = synopmob_wait_data_ready(base);
- }
- }
- }
- }
- return ret;
- }
- static int sd_send_cmd7(sdc_t* sdc)
- {
- return sd_send_cmd_r1(sdc, 0x47, sdc->rca<<16, 1);
- }
- static int sd_send_uncmd7(sdc_t* sdc)
- {
- int ret;
- unsigned int intst;
- unsigned int base = sdc->ip_base;
- ret = synopmob_execute_command(base, 0x7, 0);
- if (!ret) {
- ret = synopmob_wait_command_done(base, &intst, INTMSK_CMD_DONE);
- if (!ret) {
- synopmob_set_register(base+RINTSTS, intst);
- ret = synopmob_handle_standard_rinsts(intst);
- }
- }
- return ret;
- }
- static int sd_send_cmd16(sdc_t* sdc)
- {
- return sd_send_cmd_r1(sdc, 0x50, 512, 0);
- }
- static int sd_send_cmd55(sdc_t* sdc)
- {
- return sd_send_cmd_r1(sdc, 0x77, sdc->rca<<16, 0);
- }
- static int sd_send_acmd6(sdc_t* sdc, unsigned int bitwidth)
- {
- unsigned int cmd_arg;
- int ret;
- unsigned int base = sdc->ip_base;
- ret = sd_send_cmd55(sdc);
- if (!ret) {
- cmd_arg = 0; //default to 1bit mode
- if (bitwidth == 4) {
- cmd_arg = 2; // 4bit mode
- }
- ret = sd_send_cmd_r1(sdc, 0x2046, cmd_arg, 0);
- if (!ret) {
- if (bitwidth == 4) {
- synopmob_set_register(base+CTYPE, FOUR_BIT_MODE);
- }
- else {
- synopmob_set_register(base+CTYPE, ONE_BIT_MODE);
- }
- }
- }
- return ret;
- }
- #ifdef SDC_USE_IDMA
- static int sdc_read_write_block(HSDC handle, unsigned int rw, unsigned int blk, unsigned int num, unsigned char* buffer)
- {
- sdc_t* sdc = (sdc_t*)handle;
- volatile DmaDesc *pDmaDesc = sdc->pDmaDesc;
- int ret;
- unsigned int intsts = 0;
- unsigned int cmd;
- unsigned int multi = 0;
- unsigned int base = sdc->ip_base;
- int loop_for_command_done_check = DATA_TRANSFER_OVER_TIMEOUT_USEC;
- rt_err_t err;
- // valid check
- if (synopmob_read_register(base+CDETECT) & 1) {
- return ERRCARDNOTCONN;
- }
- if (!num || num > 16) {
- return ERRNOTSUPPORTED;
- }
- if (blk + num > sdc->sectors) {
- return ERRADDRESSRANGE;
- }
- if ( rw ) {
- flush_dcache_range((unsigned long)buffer, num << 9);
- }
- else {
- // to avoid memset bug?
- inv_dcache_range((unsigned long)buffer, num << 9);
- }
- err = rt_sem_take(sdc->mutex, RT_WAITING_FOREVER);
- if (err != RT_EOK) {
- return ERRNORES;
- }
- // reset
- synopmob_set_bits(base+CTRL, FIFO_RESET); //reset FIFO
- while (synopmob_read_register(base+CTRL) & FIFO_RESET);
- synopmob_set_register(base + RINTSTS, 0xfffe); //clear interrupts
- cmd = 0x2658; // write
- if ( !rw ) {
- cmd = 0x2251; //read
- }
- //if (num > 1) {
- if (num >= 1) { // some card fail on sigle-block mode, so use multi-block instead of sigle-block mode.
- cmd++;
- multi++;
- }
- if (sdc->card_type == SD_TYPE) {
- blk <<= 9; //SD stadand capability card use 512 unit.
- }
- num <<= 9;
-
- pDmaDesc->desc0 |= DescOwnByDma | DescFirstDesc | DescLastDesc;
- pDmaDesc->desc1 = ((num << DescBuf1SizeShift) & DescBuf1SizMsk);
- pDmaDesc->desc2 = (unsigned int)buffer;
- flush_dcache_range((unsigned long)pDmaDesc, sizeof(DmaDesc)); // add SZ_ADJUST
- synopmob_set_register(base + DBADDR, (unsigned int)(pDmaDesc)); // add SZ_ADJUST
- synopmob_set_register(base+BLKSIZ, 512);
- synopmob_set_register(base+BYTCNT, num);
- synopmob_set_bits(base + CTRL, CTRL_USE_IDMAC);
- synopmob_set_bits(base + BMOD,BMOD_DE);
-
- ret = synopmob_execute_command(base, cmd, blk);
- if ( !ret ) {
- ret = ERRIDMA;
- synopmob_set_bits(base+CTRL, INT_ENABLE);
- err = rt_sem_take(sdc->sem, DMA_TRANSFER_TIMEOUT_TICKS);
- if ( !err ) {
- while (--loop_for_command_done_check > 0) {
- intsts = synopmob_read_register(base+RINTSTS);
- if ((intsts & (INTMSK_CMD_DONE|INTMSK_DAT_OVER)) == (INTMSK_CMD_DONE|INTMSK_DAT_OVER)) {
- break;
- }
- plat_loop(1);
- }
- ret = synopmob_handle_standard_rinsts(intsts);
- if (!ret ) {
- if( !loop_for_command_done_check || !(sdc->idsts & 0x100)) { //normal interrupt
- ret = ERRIDMA;
- }
- }
- }
- }
- if (ret) {
- char* op = "read";
- if (rw)
- op = "write";
-
- SDIO_PRINTF("sdc_read_write_block(%s) fail:, ret = %d\n", op, ret);
- }
- synopmob_clear_bits(base+CTRL, INT_ENABLE);
- synopmob_clear_bits(base + CTRL, CTRL_USE_IDMAC);
- synopmob_clear_bits(base + BMOD,BMOD_DE);
- synopmob_set_register(base+BLKSIZ, 512);
- synopmob_set_register(base+BYTCNT, 512);
- synopmob_set_register(base + RINTSTS, 0xfffe);
- if ( !ret && rw) {
- ret = synopmob_wait_data_ready(base);
- }
- if (!ret && multi ) { //send STOP_TRANSACTION command
- ret = sd_send_cmd_r1(sdc, 0x404c, 0, 1);
- }
- rt_sem_release(sdc->mutex);
-
- if ( !rw && !ret ) { //read
- inv_dcache_range((unsigned long)buffer, num);
- }
-
- return ret;
- }
- #else //no IDMA
- static int sdc_read_write_block(HSDC handle, unsigned int rw, unsigned int blk, unsigned int num, unsigned char* buffer)
- {
- sdc_t* sdc = (sdc_t*)handle;
- volatile DmaDesc *pDmaDesc = sdc->pDmaDesc;
- int ret;
- unsigned int intsts = 0;
- unsigned int entries;
- unsigned int cmd;
- unsigned int multi = 0;
- unsigned int base = sdc->ip_base;
- int loop_for_command_done_check = DATA_TRANSFER_OVER_TIMEOUT_USEC;
- rt_err_t err;
- // valid check
- if (synopmob_read_register(base+CDETECT) & 1) {
- return ERRCARDNOTCONN;
- }
- if (!num || num > 16) {
- return ERRNOTSUPPORTED;
- }
- if (blk + num > sdc->sectors) {
- return ERRADDRESSRANGE;
- }
- if ( rw ) {
- flush_dcache_range((unsigned long)buffer, num << 9);
- }
- else {
- // to avoid memset bug?
- inv_dcache_range((unsigned long)buffer, num << 9);
- }
- err = rt_sem_take(sdc->mutex, RT_WAITING_FOREVER);
- if (err != RT_EOK) {
- return ERRNORES;
- }
- // reset
- synopmob_set_bits(base+CTRL, FIFO_RESET); //reset FIFO
- while (synopmob_read_register(base+CTRL) & FIFO_RESET);
- synopmob_set_register(base + RINTSTS, 0xfffe); //clear interrupts
- cmd = 0x2658; // write
- if ( !rw ) {
- cmd = 0x2251; //read
- }
- if (num > 1) {
- cmd++;
- multi++;
- }
- if (sdc->card_type == SD_TYPE) {
- blk <<= 9; //SD stadand capability card use 512 unit.
- }
- num <<= 9;
-
- synopmob_set_register(base+BLKSIZ, 512);
- synopmob_set_register(base+BYTCNT, num);
-
- ret = synopmob_execute_command(base, cmd, blk);
- if ( !ret ) {
- while (1) {
- ret = synopmob_wait_command_done(base, &intsts, 0);
- if (ret)
- break;
- ret = synopmob_handle_standard_rinsts(intsts);
- if (ret)
- break;
- if (!rw && (intsts & (INTMSK_RXDR|INTMSK_DAT_OVER)) ){
- while (num > 0 ) {
- entries = synopmob_read_register(base + STATUS);
- if (!GET_FIFO_COUNT(entries))
- break;
- *((volatile unsigned int*)buffer) = synopmob_read_register(base + FIFODAT);
- buffer += 4;
- num -= 4;
- }
- }
-
- if (rw && ( intsts & INTMSK_TXDR ) ) {
- while (num > 0) {
- entries = synopmob_read_register(base+STATUS);
- if ( entries & 8 ) { //FIFO is full
- break;
- }
- synopmob_set_register(base+FIFODAT, *((volatile unsigned int*)buffer));
- buffer += 4;
- num -= 4;
- }
- }
-
- if ( intsts & INTMSK_DAT_OVER ) {
- break;
- }
- if (intsts & INTMSK_CMD_DONE) {
- entries = synopmob_read_register(base+RESP0);
- ret = synopmob_check_r1_resp(entries);
- if (ret) {
- break;
- }
- }
- synopmob_set_register(base+RINTSTS, intsts); //write to clear
- intsts = 0;
- }
-
- if (intsts) {
- synopmob_set_register(base+RINTSTS, intsts); //write to clear
- }
- }
- synopmob_set_register(base+BLKSIZ, 512);
- synopmob_set_register(base+BYTCNT, 512);
- synopmob_set_register(base + RINTSTS, 0xfffe);
- if ( !ret && rw) {
- ret = synopmob_wait_data_ready(base);
- }
- if (!ret && multi ) { //send STOP_TRANSACTION command
- ret = sd_send_cmd_r1(sdc, 0x404c, 0, 1);
- }
- rt_sem_release(sdc->mutex);
-
- if ( !rw && !ret ) { //read
- inv_dcache_range((unsigned long)buffer, num);
- }
-
- return ret;
- }
- #endif //SDC_USE_IDMA
- int sdc_write_block(HSDC handle, unsigned int blk, unsigned int num, unsigned char* buffer)
- {
- return sdc_read_write_block(handle, 1, blk, num, buffer);
- }
- int sdc_read_block(HSDC handle, unsigned int blk, unsigned int num, unsigned char* buffer)
- {
- return sdc_read_write_block(handle, 0, blk, num, buffer);
- }
- int sdc_erase_block(HSDC handle, unsigned int blk, unsigned int num)
- {
- int ret;
- sdc_t* sdc = (sdc_t*)handle;
-
- if (sdc->card_type == SD_TYPE) {
- blk <<= 9; //SD stadand capability card use 512 unit.
- num = ((num-1)<<9) + blk;
- }
- else {
- num = blk + num - 1;
- }
-
- ret = sd_send_cmd_r1(sdc, 0x40|32, blk, 0); // cmd32
- if (!ret) {
- ret = sd_send_cmd_r1(sdc, 0x40|33, num, 0); // cmd33
- if (!ret) {
- ret = sd_send_cmd_r1(sdc, 0x40|38, 0, 1); // cmd38
- }
- }
- return ret;
- }
- int sdc_get_sector_num(HSDC handle)
- {
- return ((sdc_t*)handle)->sectors;
- }
- static int sd_send_cmd9(sdc_t* sdc)
- {
- int ret;
- unsigned int intst;
- unsigned int resp0;
- unsigned int resp1;
- unsigned int resp2;
- unsigned int resp3;
- unsigned int base = sdc->ip_base;
- unsigned int C_SIZE;
- unsigned int C_SIZE_MULT;
- unsigned int READ_BL_LEN;
- ret = synopmob_execute_command(base, 0xC9, sdc->rca<<16);
- if (!ret) {
- ret = synopmob_wait_command_done(base, &intst, INTMSK_CMD_DONE);
- if (!ret) {
- synopmob_set_register(base+RINTSTS, intst);
- ret = synopmob_handle_standard_rinsts(intst);
- if (!ret) {
- sdc->csd[0] = resp0 = synopmob_read_register(base+RESP0);
- sdc->csd[1] = resp1 = synopmob_read_register(base+RESP1);
- sdc->csd[2] = resp2 = synopmob_read_register(base+RESP2);
- sdc->csd[3] = resp3 = synopmob_read_register(base+RESP3);
-
- if ((resp3>>30) == 0) { //CSD version 1.0
- C_SIZE = (resp1 >> 30) | ((resp2 & 0x3ff)<<2);
- C_SIZE_MULT = ((resp1 >> 15) & 0x07);
- READ_BL_LEN = ((resp2 >> 16) & 0xf);
- sdc->sectors = ((((C_SIZE+1)<<(C_SIZE_MULT+2))<<(READ_BL_LEN))>>9);
- }
- else { //CSD version 2.0
- sdc->sectors = (((resp1 >> 16)+1)<<10);
- }
- }
- }
- }
- return ret;
- }
- static int sd_send_cmd5(sdc_t* sdc, unsigned int arg, unsigned int* resp)
- {
- unsigned int cmd_reg = 0x45;
- unsigned int intst;
- int ret;
- unsigned int base = sdc->ip_base;
- ret = synopmob_execute_command(base, cmd_reg, arg);
- if (!ret) {
- ret = synopmob_wait_command_done(base, &intst, INTMSK_CMD_DONE);
- if (!ret) {
- synopmob_set_register(base+RINTSTS, intst);
- ret = synopmob_handle_standard_rinsts(intst);
- if (!ret) {
- *resp = synopmob_read_register(base+RESP0);
- }
- }
- }
-
- return ret;
- }
- static int sd_send_cmd8(sdc_t* sdc)
- {
- int ret;
- unsigned int cmd_reg = 0x48;
- unsigned int intst;
- unsigned int err = 0;
- unsigned int base = sdc->ip_base;
-
- ret = synopmob_execute_command(base, cmd_reg, 0x000001A5);
- if (!ret) {
- while (1) {
- ret = synopmob_wait_command_done(base, &intst, 0);
- if (!ret) {
- synopmob_set_register(base+RINTSTS, intst);
- err |= synopmob_handle_standard_rinsts(intst);
- if (intst & INTMSK_CMD_DONE) {
- break;
- }
- }
- }
- }
- return err;
- }
- static int sd_send_acmd41(sdc_t* sdc, int* hcs)
- {
- unsigned int cmd_reg = 0x69;
- unsigned int resp;
- int ret = 0;
- unsigned int count = ACMD41_RETRY_COUNT;
- unsigned int cmd_arg = 0xff8000;
- unsigned int base = sdc->ip_base;
- if (*hcs) {
- cmd_arg |= (1<<30);
- }
- while ( count > 0) {
- SDC_WHERE();
- ret = sd_send_cmd55(sdc);
- if (ret)
- break;
- SDC_WHERE();
- ret = synopmob_execute_command(base, cmd_reg, cmd_arg);
- if (ret)
- break;
- SDC_WHERE();
- ret = synopmob_wait_command_done(base, &resp, INTMSK_CMD_DONE);
- if ( ret )
- break;
- SDC_WHERE();
- synopmob_set_register(base+RINTSTS, resp);
- ret = synopmob_handle_standard_rinsts(resp);
- if (!ret) {
- SDC_WHERE();
- resp = synopmob_read_register(base+RESP0);
- if (resp & 0x80000000) { //card is ready.
- SDC_WHERE();
- if ( !(resp & (1<<30)) ) {
- SDC_WHERE();
- *hcs = 0;
- }
- if ( (resp & 0x00ff8000) != 0x00ff8000 ) { //not supported voltage
- ret = ERRHARDWARE;
- }
- break;
- }
- }
- --count;
- plat_loop(1);
- }
- if (!count)
- ret = ERRACMD41TIMEOUT;
-
- return ret;
- }
- static int sd_send_acmd51(sdc_t* sdc) //Send SCR
- {
- unsigned int cmd_reg = 0x2273;
- unsigned int resp;
- int ret;
- unsigned int intst = 0;
- unsigned int entries;
- int count = 1;
- unsigned int base = sdc->ip_base;
- ret = sd_send_cmd55(sdc);
- if (!ret) {
- synopmob_set_register(base+BLKSIZ, 8);
- synopmob_set_register(base+BYTCNT, 8);
- ret = synopmob_execute_command(base, cmd_reg, 0);
- if (!ret) {
- while (1) {
- ret = synopmob_wait_command_done(base, &intst, 0);
- if (ret) {
- break;
- }
- ret = synopmob_handle_standard_rinsts(intst);
- if (ret) {
- break;
- }
-
- if (intst & INTMSK_CMD_DONE) {
- resp = synopmob_read_register(base+RESP0);
- ret = synopmob_check_r1_resp(resp);
- if (ret)
- break;
- }
- if (intst & INTMSK_DAT_OVER) {
- entries = synopmob_read_register(base + STATUS);
- if (GET_FIFO_COUNT(entries) == 2) {
- while (count >= 0) {
- entries = synopmob_read_register(base + FIFODAT);
- sdc->scr[count--] = BE32_TO_CPU(entries);
- }
- }
- break;
- }
- synopmob_set_register(base+RINTSTS, intst);
- intst = 0;
- }
- if (intst) {
- synopmob_set_register(base+RINTSTS, intst);
- }
- }
- synopmob_set_register(base+BLKSIZ, 512);
- synopmob_set_register(base+BYTCNT, 512);
- }
- return ret;
- }
- static int sd_send_cmd6(sdc_t* sdc, unsigned int cmd_arg, unsigned int* data_buff)
- {
- unsigned int cmd_reg = 0x2246;
- unsigned int resp;
- int ret;
- unsigned int intst = 0;
- unsigned int entries;
- int count = 64;
- unsigned int base = sdc->ip_base;
- synopmob_set_register(base+BLKSIZ, 64);
- synopmob_set_register(base+BYTCNT, 64);
- ret = synopmob_execute_command(base, cmd_reg, cmd_arg);
- if (!ret) {
- while (1) {
- ret = synopmob_wait_command_done(base, &intst, 0);
- if (ret) {
- break;
- }
- ret = synopmob_handle_standard_rinsts(intst);
- if (ret) {
- break;
- }
-
- if (intst & INTMSK_CMD_DONE) {
- resp = synopmob_read_register(base+RESP0);
- ret = synopmob_check_r1_resp(resp);
- if (ret)
- break;
- }
- while (count > 0) {
- entries = synopmob_read_register(base + STATUS);
- if ( !GET_FIFO_COUNT(entries) ) {
- break;
- }
- *(data_buff++) = synopmob_read_register(base + FIFODAT);
- count -= 4;
- }
- if (intst & INTMSK_DAT_OVER) {
- break;
- }
- synopmob_set_register(base+RINTSTS, intst); //write to clear
- intst = 0;
- }
- if (intst) {
- synopmob_set_register(base+RINTSTS, intst); //write to clear
- }
- }
- synopmob_set_register(base+BLKSIZ, 512);
- synopmob_set_register(base+BYTCNT, 512);
- return ret;
- }
- static int synopmob_send_clock_only_cmd(unsigned int base)
- {
- return synopmob_execute_command(base, 0x202000, 0);
- }
- static int synopmob_disable_all_clocks(unsigned int base)
- {
- synopmob_set_register(base+CLKENA, 0);
- return synopmob_send_clock_only_cmd(base);
- }
- static int synopmob_enable_clocks_with_val(unsigned int base, unsigned int val)
- {
- synopmob_set_register(base+CLKENA, val);
- return synopmob_send_clock_only_cmd(base);
- }
- static int synopmob_set_clk_freq(sdc_t* sdc, unsigned int divider)
- {
- #define MAX_DIVIDER_VALUE 0xff
-
- unsigned int orig_clkena;
- int retval;
- unsigned int base = sdc->ip_base;
- if (divider > MAX_DIVIDER_VALUE) {
- return 0xffffffff;
- }
- /* To make sure we dont disturb enable/disable settings of the cards*/
- orig_clkena = synopmob_read_register(base+CLKENA);
- /* Disable all clocks before changing frequency the of card clocks */
- if ((retval = synopmob_disable_all_clocks(base)) != 0) {
- return retval;
- }
- /* Program the clock divider in our case it is divider 0 */
- synopmob_clear_bits(base+CLKDIV, MAX_DIVIDER_VALUE);
- synopmob_set_bits(base+CLKDIV, divider);
-
- /*Send the command to CIU using synopmob_send_clock_only_cmd and enable the clocks in CLKENA register */
- if ((retval = synopmob_send_clock_only_cmd(base)) != 0) {
- synopmob_enable_clocks_with_val(base, orig_clkena);
- return retval;
- }
- return synopmob_enable_clocks_with_val(base, orig_clkena);
- }
- static int enum_sd_card(sdc_t* sdc)
- {
- int ret;
- int count = 1000;
- int hcs = 0;
- unsigned int buffer[16];
- unsigned int base = sdc->ip_base;
- if (synopmob_read_register(base+CDETECT) & 1) {
- return ERRCARDNOTCONN;
- }
-
- #if 0
- synopmob_set_bits(0x98500004, (1<<24)); //set to output mode
- synopmob_set_bits(0x98500000, (1<<24)); //power off
- plat_loop(1000000/5); //Lets give some ramp down period
- synopmob_clear_bits(0x98500000, (1<<24)); //power on
- plat_loop(1000000/5);//Lets give some ramp down period
- #endif
-
- synopmob_set_register(base+CTYPE, ONE_BIT_MODE);
- synopmob_set_register(base+CLKENA, 0x00000001); /*enable clock, non-low-power mode*/
- ret = synopmob_set_clk_freq(sdc, MMC_FOD_DIVIDER_VALUE);
-
- if ( !ret ) {
- plat_loop(1000); //enough for 74 clock.
- SDC_WHERE();
- ret = sd_send_cmd0(sdc); //CMD0 has no response
- }
-
- if ( !ret ) {
- SDC_WHERE();
- ret = sd_send_cmd8(sdc); //even if CMD8 get response, it may be V1.0 card.
- if (!ret) {
- hcs = 1;
- }
- SDC_WHERE();
- ret = sd_send_acmd41(sdc, &hcs);
- }
- if (!ret) {
- SDC_WHERE();
- ret = sd_send_cmd2(sdc); //CID
- }
- if (!ret) {
- SDC_WHERE();
- ret = sd_send_cmd3(sdc); //get RCA
- }
- if (!ret) {
- SDC_WHERE();
- ret = sd_send_cmd9(sdc); //CSD
- }
- if (!ret) {
- SDC_WHERE();
- ret = sd_send_cmd7(sdc); //select the card
- }
- if (!ret && (sdc->wkmod & SDC_WKMOD_4WIRE) ) {
- SDC_WHERE();
- ret = sd_send_acmd51(sdc); //SCR
- if (!ret && (sdc->scr[1] & 0x00040000)) { // 4bit mode supported?
- ret = sd_send_acmd6(sdc, 4); //switch to 4bit mode
- }
- }
-
- if (!ret && (sdc->wkmod & SDC_WKMOD_50M_HI_SPEED) && (sdc->csd[2] & 0x40000000) ) { //judge whether class10 is supported? CMD6 is belonging to class10.
- SDC_WHERE();
- ret = sd_send_cmd6(sdc, 0x00fffff1, buffer); //switch to high speed mode.
- if ( !ret && (*(((unsigned char*)buffer)+13)&0x02) ) { //the card support high speed mode?
- SDC_WHERE();
- ret = sd_send_cmd6(sdc, 0x80fffff1, buffer); //switch to high speed mode.
- if (!ret && ((*(((unsigned char*)buffer)+16) & 0xf) == 1) ) {
- //switch to high speed mode sucess.
- sd_send_uncmd7(sdc); //deselect the card
- sd_send_cmd9(sdc); //CSD
- ret = sd_send_cmd7(sdc); //select the card
- }
- }
- }
-
- if (!ret && (sdc->wkmod & (SDC_WKMOD_50M_HI_SPEED|SDC_WKMOD_25M_STAND_SPEED))) {
- if ( (sdc->csd[3] & 0xff) == 0x5A ) { //50MHz high speed mode.
- SDC_WHERE();
- ret = synopmob_set_clk_freq(sdc, (((CIU_CLK)/(50000*2))));
- }
- else if ( (sdc->csd[3] & 0xff) == 0x32 ) {
- SDC_WHERE(); //25MHz standard speed mode.
- ret = synopmob_set_clk_freq(sdc, sdc_clk_divider/*ONE_BIT_BUS_FREQ*/);
- }
- }
- if (!ret) {
- sdc->card_type = SD_TYPE;
- if (hcs) {
- sdc->card_type = SD_2_0_TYPE;
- }
- }
- return ret;
- }
- int sdio_drv_creg_read(HSDC handle, int addr, int fn, unsigned int *resp)
- {
- sdc_t* sdc = (sdc_t*)handle;
- unsigned int arg;
- unsigned int cmd_reg = 0x74;
- unsigned int intst;
- int ret;
- unsigned int base = sdc->ip_base;
- rt_err_t err;
- if(resp) {
- *resp = 0;
- }
-
- err = rt_sem_take(sdc->mutex, RT_WAITING_FOREVER);
- if (err != RT_EOK) {
- return ERRNORES;
- }
- arg = (fn << 28) | (addr << 9);
- ret = synopmob_execute_command(base, cmd_reg, arg);
- if (!ret) {
- ret = synopmob_wait_command_done(base, &intst, INTMSK_CMD_DONE);
- if (!ret) {
- synopmob_set_register(base+RINTSTS, intst); //write to clear
- ret = synopmob_handle_standard_rinsts(intst);
- if (!ret) {
- *resp = synopmob_read_register(base+RESP0);
- ret = synopmob_check_r5_resp(*resp);
- }
- }
- }
- rt_sem_release(sdc->mutex);
-
- if (ret) {
- ret++;
- ret--;
- SDIO_PRINTF("sdio_drv_creg_read fail:, ret = %d\n", ret);
- }
-
- return ret;
- }
- int sdio_drv_creg_write(HSDC handle, int addr, int fn, unsigned char data, unsigned int *resp)
- {
- sdc_t* sdc = (sdc_t*)handle;
- unsigned int arg;
- unsigned int cmd_reg = 0x74;
- unsigned int intst;
- int ret;
- unsigned int base = sdc->ip_base;
- rt_err_t err;
- err = rt_sem_take(sdc->mutex, RT_WAITING_FOREVER);
- if (err != RT_EOK) {
- return ERRNORES;
- }
-
- arg = (1 << 31) | (fn << 28) | (1 << 27) | (addr << 9) | data;
- ret = synopmob_execute_command(base, cmd_reg, arg);
- if (!ret) {
- ret = synopmob_wait_command_done(base, &intst, INTMSK_CMD_DONE);
- if (!ret) {
- synopmob_set_register(base+RINTSTS, intst); //write to clear
- ret = synopmob_handle_standard_rinsts(intst);
- if (!ret) {
- *resp = synopmob_read_register(base+RESP0);
- ret = synopmob_check_r5_resp(*resp);
- }
- }
- }
- rt_sem_release(sdc->mutex);
- if (ret) {
- ret++;
- ret--;
- SDIO_PRINTF("sdio_drv_creg_write fail:, ret = %d\n", ret);
- }
-
- return ret;
- }
- #define ARC_REG_DC_IVDL 0x4A
- #define ARC_REG_DC_FLDL 0x4C
- #define ARC_REG_DC_CTRL 0x48
- #define FS_FLAG 0x100
- #define _psp_get_aux(aux_reg) \
- (unsigned long)_lr((unsigned)(aux_reg))
- #define _psp_set_aux(aux_reg,value) \
- _sr((unsigned)(value),(unsigned)(aux_reg))
- extern void mmu_clean_dcache(rt_uint32_t buffer, rt_uint32_t size);
- extern void mmu_invalidate_dcache(rt_uint32_t buffer, rt_uint32_t size);
- void inv_dcache_range(unsigned long start, unsigned long len)
- {
- mmu_invalidate_dcache(start, len);
- }
- void flush_dcache_range(unsigned long start, unsigned long len)
- {
- mmu_clean_dcache(start, len);
- }
- int g_use_bcm43362 = 0;
- static int sdio_drv_read_write(sdc_t* sdc, unsigned int rw, unsigned int addr, unsigned int fn, unsigned int bcnt,
- unsigned int bsize, unsigned char *buf)
- {
- volatile DmaDesc *pDmaDesc = sdc->pDmaDesc;
- int ret;
- unsigned int intsts = 0;
- unsigned int cmd = 0x2275;
- unsigned int base = sdc->ip_base;
- unsigned int arg;
- unsigned int num;
- int loop_for_command_done_check = 10000;//DATA_TRANSFER_OVER_TIMEOUT_USEC;
- rt_err_t err;
- //SDIO_PRINTF("------- %s, line %d buf = %08x size = %d -------\n", __FUNCTION__, __LINE__, buf, bsize);
- arg = (fn << 28) | (addr << 9);
- if (g_use_bcm43362) {
- arg |= (1 << 26); //OPcode = 1............, for AP6181.
- }
- if (bcnt == 1 && bsize <= 512)
- arg |= (bsize & 0x1ff);
- else
- arg |= ((1 << 27) | bcnt);
- if ( rw ) {
- cmd |= 0x400;
- arg |= (1 << 31);
- }
- num = bsize*bcnt;
- if ( rw ) {
- flush_dcache_range((unsigned long)buf, num);
- }
- else {
- inv_dcache_range((unsigned long)buf, num);
- }
- err = rt_sem_take(sdc->mutex, RT_WAITING_FOREVER);
- if (err != RT_EOK) {
- return ERRNORES;
- }
- //synopmob_set_bits(base+FIFOTH, 0x2 << 28);
- // reset
- synopmob_set_bits(base+CTRL, FIFO_RESET); //reset FIFO
- while (synopmob_read_register(base+CTRL) & FIFO_RESET);
- synopmob_set_register(base + RINTSTS, 0xfffe); //clear interrupts
- //pDmaDesc->desc0 = 0;
- pDmaDesc->desc0 |= DescOwnByDma | DescFirstDesc | DescLastDesc;
- pDmaDesc->desc1 = ((num << DescBuf1SizeShift) & DescBuf1SizMsk);
- pDmaDesc->desc2 = (unsigned int)buf;
- //pDmaDesc->desc3 = 0;
- flush_dcache_range((unsigned int)pDmaDesc, sizeof(DmaDesc));
- synopmob_set_register(base+DBADDR, (unsigned int)pDmaDesc);
- synopmob_set_register(base+BLKSIZ, bsize);
- synopmob_set_register(base+BYTCNT, num);
- synopmob_set_bits(base + CTRL, CTRL_USE_IDMAC);
- synopmob_set_bits(base + BMOD,BMOD_DE);
- //SDIO_PRINTF("pDmaDesc = %08x, %08x / %08x / %08x / %08x\n", pDmaDesc, pDmaDesc->desc0, pDmaDesc->desc1, pDmaDesc->desc2, pDmaDesc->desc3);
- ret = synopmob_execute_command(base, cmd, arg);
- if ( !ret ) {
- ret = ERRIDMA;
- err = rt_sem_take(sdc->sem, DMA_TRANSFER_TIMEOUT_TICKS);
- if ( !err ) {
- while (--loop_for_command_done_check > 0) {
- intsts = synopmob_read_register(base+RINTSTS);
- if ((intsts & (INTMSK_CMD_DONE|INTMSK_DAT_OVER)) == (INTMSK_CMD_DONE|INTMSK_DAT_OVER)) {
- break;
- }
- plat_loop(1);
- }
- ret = synopmob_handle_standard_rinsts(intsts);
- if (!ret ) {
- if( !loop_for_command_done_check || !(sdc->idsts & 0x100)) { //normal interrupt
- SDIO_PRINTF("------- %s, line %d idsts = %08x check = %d -------\n", __FUNCTION__, __LINE__, sdc->idsts, loop_for_command_done_check);
- ret = ERRIDMA;
- }
- }
- else
- SDIO_PRINTF("------- %s, line %d intsts = %08x buf = %08x -------\n", __FUNCTION__, __LINE__, intsts, buf);
- }
- }
- if (!ret) {
- synopmob_set_register(base + RINTSTS, 0xfffe); //clear interrupts
- synopmob_clear_bits(base + CTRL, CTRL_USE_IDMAC);
- synopmob_clear_bits(base + BMOD,BMOD_DE);
- synopmob_set_register(base+BLKSIZ, 512);
- synopmob_set_register(base+BYTCNT, 512);
- }
- else {
- char* op = "read";
- if (rw)
- op = "write";
-
- SDIO_PRINTF("sdio_drv_read_write1(%s) fail:, ret = %d\n", op, ret);
- }
-
- if ( rw && !ret ) { //write
- ret = synopmob_wait_data_ready(base);
- }
- rt_sem_release(sdc->mutex);
- if ( !rw && !ret ) { //read
- inv_dcache_range((unsigned long)buf, num);
- }
-
- if (ret) {
- char* op = "read";
- if (rw)
- op = "write";
-
- SDIO_PRINTF("sdio_drv_read_write2(%s) fail:, ret = %d\n", op, ret);
- }
- return ret;
- //return ret ? 0/*false*/ : 1/*true*/;
- }
- int sdio_drv_read(HSDC handle, unsigned int addr, unsigned int fn, unsigned int bcnt,
- unsigned int bsize, unsigned char *buf)
- {
- return sdio_drv_read_write((sdc_t*)handle, 0, addr, fn, bcnt, bsize, buf);
- }
- int sdio_drv_write(HSDC handle, unsigned int addr, unsigned int fn, unsigned int bcnt,
- unsigned int bsize, unsigned char *buf)
- {
- return sdio_drv_read_write((sdc_t*)handle, 1, addr, fn, bcnt, bsize, buf);
- }
- static void dumpchain(DmaDesc *pChain)
- {
- int i = 0;
- DmaDesc *tmp_pChain = pChain;
-
- while(tmp_pChain && i < 10)
- {
- SDIO_PRINTF("[%d]: chain =%p, buf = %p, size = %d, csi = %08x, next = %p\n", i, tmp_pChain, (DmaDesc *)tmp_pChain->desc2, tmp_pChain->desc1, tmp_pChain->desc0, (DmaDesc *)tmp_pChain->desc3);
- tmp_pChain = (DmaDesc *)tmp_pChain->desc3;
- i++;
- if(tmp_pChain == pChain)
- break;
- }
- }
- #ifdef DONT_COPY_NET_PAYLOAD_TO_SEND
- #if 1
- int sdio_drv_chain_write(sdc_t* sdc, unsigned int addr, unsigned int fn, unsigned int bcnt, unsigned int bsize, buf_chain_t *chain)
- {
- int ret;
- unsigned int intsts = 0;
- unsigned int cmd = 0x2275;
- unsigned int base = sdc->ip_base;
- unsigned int arg;
- unsigned int num;
- unsigned int chain_len = 0;
- int loop_for_command_done_check = DATA_TRANSFER_OVER_TIMEOUT_USEC;
- rt_err_t err;
- unsigned int rw = 1;
- DmaDesc *tmpDesc = (DmaDesc *)chain;
- DmaDesc *lastDesc = (void*)0;
- arg = (fn << 28) | (addr << 9);
- if (bcnt == 1 && bsize <= 512)
- arg |= (bsize & 0x1ff);
- else
- arg |= ((1 << 27) | bcnt);
- if ( rw ) {
- cmd |= 0x400;
- arg |= (1 << 31);
- }
- num = bsize*bcnt;
- err = rt_sem_take(sdc->mutex, RT_WAITING_FOREVER);
- if (err != RT_EOK) {
- return ERRNORES;
- }
-
- // reset
- synopmob_set_bits(base+CTRL, FIFO_RESET); //reset FIFO
- while (synopmob_read_register(base+CTRL) & FIFO_RESET);
- synopmob_set_register(base + RINTSTS, 0xfffe); //clear interrupts
-
- while(tmpDesc != 0) {
- // make sure size is little than DescBuf1SizMsk
- if(tmpDesc->desc1 > (DescBuf1SizMsk >> DescBuf1SizeShift)) {
- // TBD... fix me
- rt_sem_release(sdc->mutex);
- return 0;
- }
- // TBD... fix me, we must align tmpDesc->desc2 to 4 ?
-
- tmpDesc->desc0 = DescOwnByDma | DescSecAddrChained;
-
- // is it last node?
- if(tmpDesc->desc3 == 0 || tmpDesc->desc3 == (unsigned int)chain) {
- tmpDesc->desc0 |= DescLastDesc;
- lastDesc = tmpDesc;
- }
- else {
- tmpDesc->desc0 |= DescDisInt; //disable interrupt...
- }
-
- // is it first node?
- if((char *)tmpDesc == (char *)chain) {
- tmpDesc->desc0 |= DescFirstDesc;
- }
- flush_dcache_range(tmpDesc->desc2, tmpDesc->desc1);
-
- tmpDesc = (DmaDesc *)tmpDesc->desc3;
- chain_len += sizeof(buf_chain_t);
- if((char *)tmpDesc == (char *)chain) {
- break;
- }
- }
- lastDesc->desc3 = (unsigned int)chain;
- //FIXME, chain must be continuous arrry.
- flush_dcache_range((unsigned long)chain, chain_len);
- synopmob_set_register(base+DBADDR, (unsigned int)(chain));
-
- synopmob_set_register(base+BLKSIZ, bsize);
- synopmob_set_register(base+BYTCNT, num);
- synopmob_set_bits(base + CTRL, CTRL_USE_IDMAC);
- synopmob_set_bits(base + BMOD,BMOD_DE);
-
- ret = synopmob_execute_command(base, cmd, arg);
- if ( !ret ) {
- ret = ERRIDMA;
- err = rt_sem_take(sdc->sem, DMA_TRANSFER_TIMEOUT_TICKS);
- if ( !err ) {
- while (--loop_for_command_done_check > 0) {
- intsts = synopmob_read_register(base+RINTSTS);
- if ((intsts & (INTMSK_CMD_DONE|INTMSK_DAT_OVER)) == (INTMSK_CMD_DONE|INTMSK_DAT_OVER)) {
- break;
- }
- plat_loop(1);
- }
- ret = synopmob_handle_standard_rinsts(intsts);
- if (!ret ) {
- if( !loop_for_command_done_check || !(sdc->idsts & 0x100)) { //normal interrupt
- ret = ERRIDMA;
- }
- }
- }
- }
- if (!ret) {
- synopmob_set_register(base + RINTSTS, 0xfffe); //clear interrupts
- synopmob_clear_bits(base + CTRL, CTRL_USE_IDMAC);
- synopmob_clear_bits(base + BMOD,BMOD_DE);
- synopmob_set_register(base+BLKSIZ, 512);
- synopmob_set_register(base+BYTCNT, 512);
- }
- else {
- char* op = "read";
- if (rw)
- op = "write";
-
- SDIO_PRINTF("sdio_drv_chain_write1(%s) fail:, ret = %d, bsize = %d * %d\n", op, ret, bsize, bcnt);
- dumpchain((DmaDesc *)chain);
- }
- if ( rw && !ret ) {
- ret = synopmob_wait_data_ready(base);
- }
- rt_sem_release(sdc->mutex);
- if (ret) {
- ret++;
- ret--;
- SDIO_PRINTF("sdio_drv_chain_write2, fail:, ret = %d\n", ret);
- }
- return ret;
- //return ret ? 0/*false*/ : 1/*true*/;
- }
- #elif 0
- int sdio_drv_chain_write(sdc_t* sdc, unsigned int addr, unsigned int fn, unsigned int bcnt, unsigned int bsize, buf_chain_t *chain)
- {
- //static volatile DmaDesc __attribute__ ((aligned(32))) st_pchain[4];
- volatile DmaDesc *st_pchain = (DmaDesc *)0x9a700000;
-
- int ret;
- unsigned int intsts = 0;
- unsigned int cmd = 0x2275;
- unsigned int base = sdc->ip_base;
- unsigned int arg;
- unsigned int num;
- unsigned int chain_len = 0;
- int loop_for_command_done_check = DATA_TRANSFER_OVER_TIMEOUT_USEC;
- char err;
- unsigned int rw = 1;
- buf_chain_t *usrchain;
- unsigned int desc0;
- unsigned int length = 0;
- if (!chain) {
- return 0;
- }
- arg = (fn << 28) | (addr << 9);
- if (bcnt == 1 && bsize <= 512)
- arg |= (bsize & 0x1ff);
- else
- arg |= ((1 << 27) | bcnt);
- if ( rw ) {
- cmd |= 0x400;
- arg |= (1 << 31);
- }
- num = bsize*bcnt;
- OSSemPend(sdc->mutex, 0, &err);
- if (err != OS_NO_ERR) {
- return ERRNORES;
- }
-
- // reset
- synopmob_set_bits(base+CTRL, FIFO_RESET); //reset FIFO
- while (synopmob_read_register(base+CTRL) & FIFO_RESET);
- synopmob_set_register(base + RINTSTS, 0xfffe); //clear interrupts
- usrchain = chain;
- while (1) {
- if(usrchain->size > (DescBuf1SizMsk >> DescBuf1SizeShift)) {
- // TBD... fix me
- OSSemPost (sdc->mutex);
- return 0;
- }
- length += usrchain->size;
- desc0 = DescOwnByDma | DescSecAddrChained;
- if (!usrchain->next || usrchain->next == chain) {
- desc0 |= DescLastDesc;
- }
- else {
- desc0 |= DescDisInt; //disable interrupt...
- }
- if(usrchain == chain) {
- desc0 |= DescFirstDesc;
- }
- st_pchain[chain_len].desc0 = desc0;
- st_pchain[chain_len].desc1 = (unsigned int)usrchain->size;
- st_pchain[chain_len].desc2 = (unsigned int)usrchain->buf;
- st_pchain[chain_len].desc3 = (unsigned int)(&st_pchain[chain_len+1]);
- flush_dcache_range((unsigned int)usrchain->buf, usrchain->size);
-
- usrchain = usrchain->next;
- if( !usrchain || usrchain == chain) {
- break;
- }
- if (++chain_len >= 4) {
- while(1) SDIO_PRINTF("sdio_drv_chain_write:long chain!\n");
- }
- }
- st_pchain[chain_len].desc3 = (unsigned int)(&st_pchain[0]);
- if (length != num) {
- while (1) SDIO_PRINTF("sdio_drv_chain_write:too long packet!\n");
- }
- synopmob_set_register(base+DBADDR, (unsigned int)(st_pchain));
-
- synopmob_set_register(base+BLKSIZ, bsize);
- synopmob_set_register(base+BYTCNT, num);
- synopmob_set_bits(base + CTRL, CTRL_USE_IDMAC);
- synopmob_set_bits(base + BMOD,BMOD_DE);
-
- ret = synopmob_execute_command(base, cmd, arg);
- if ( !ret ) {
- ret = ERRIDMA;
- OSSemPend(sdc->sem, DMA_TRANSFER_TIMEOUT_TICKS, &err);
- if ( !err ) {
- while (--loop_for_command_done_check > 0) {
- intsts = synopmob_read_register(base+RINTSTS);
- if ((intsts & (INTMSK_CMD_DONE|INTMSK_DAT_OVER)) == (INTMSK_CMD_DONE|INTMSK_DAT_OVER)) {
- break;
- }
- plat_loop(1);
- }
- ret = synopmob_handle_standard_rinsts(intsts);
- if (!ret ) {
- if( !loop_for_command_done_check || !(sdc->idsts & 0x100)) { //normal interrupt
- ret = ERRIDMA;
- }
- }
- }
- }
- if (!ret) {
- synopmob_set_register(base + RINTSTS, 0xfffe); //clear interrupts
- synopmob_clear_bits(base + CTRL, CTRL_USE_IDMAC);
- synopmob_clear_bits(base + BMOD,BMOD_DE);
- synopmob_set_register(base+BLKSIZ, 512);
- synopmob_set_register(base+BYTCNT, 512);
- }
- else {
- char* op = "read";
- if (rw)
- op = "write";
-
- SDIO_PRINTF("sdio_drv_chain_write1(%s) fail:, ret = %d\n", op, ret);
- }
- if ( rw && !ret ) {
- ret = synopmob_wait_data_ready(base);
- }
- OSSemPost (sdc->mutex);
- return ret;
- //return ret ? 0/*false*/ : 1/*true*/;
- }
- #else
- static unsigned char __attribute__ ((aligned(32))) st_net_buf[2*1024];
- int sdio_drv_chain_write(sdc_t* sdc, unsigned int addr, unsigned int fn, unsigned int bcnt, unsigned int bsize, buf_chain_t *chain)
- {
- static volatile DmaDesc __attribute__ ((aligned(32))) st_pchain;
-
- int ret;
- unsigned int intsts = 0;
- unsigned int cmd = 0x2275;
- unsigned int base = sdc->ip_base;
- unsigned int arg;
- unsigned int num;
- int loop_for_command_done_check = DATA_TRANSFER_OVER_TIMEOUT_USEC;
- rt_err_t err;
- unsigned int rw = 1;
- buf_chain_t *usrchain;
- unsigned int length = 0;
- if (!chain) {
- return 0;
- }
- arg = (fn << 28) | (addr << 9);
- if (bcnt == 1 && bsize <= 512)
- arg |= (bsize & 0x1ff);
- else
- arg |= ((1 << 27) | bcnt);
- if ( rw ) {
- cmd |= 0x400;
- arg |= (1 << 31);
- }
- num = bsize*bcnt;
- err = rt_sem_take(sdc->mutex, RT_WAITING_FOREVER);
- if (err != RT_EOK) {
- return ERRNORES;
- }
-
-
- // reset
- synopmob_set_bits(base+CTRL, FIFO_RESET); //reset FIFO
- while (synopmob_read_register(base+CTRL) & FIFO_RESET);
- synopmob_set_register(base + RINTSTS, 0xfffe); //clear interrupts
- usrchain = chain;
- while (1) {
- if (length + usrchain->size >= sizeof(st_net_buf)) {
- while(1) SDIO_PRINTF("too long net pkt\n");
- }
-
- memcpy(st_net_buf + length, usrchain->buf, usrchain->size);
- length += usrchain->size;
- usrchain = usrchain->next;
- if (!usrchain || usrchain->next == chain) {
- break;
- }
- }
-
- st_pchain.desc0 = DescOwnByDma | DescSecAddrChained | DescLastDesc | DescFirstDesc;
- st_pchain.desc1 = length;
- st_pchain.desc2 = (unsigned int)st_net_buf;
- st_pchain.desc3 = (unsigned int)&st_pchain;
- flush_dcache_range((unsigned long)st_net_buf, length);
-
- if (length != num) {
- while (1) SDIO_PRINTF("sdio_drv_chain_write:too long packet!\n");
- }
- synopmob_set_register(base+DBADDR, (unsigned int)(&st_pchain));
-
- synopmob_set_register(base+BLKSIZ, bsize);
- synopmob_set_register(base+BYTCNT, num);
- synopmob_set_bits(base + CTRL, CTRL_USE_IDMAC);
- synopmob_set_bits(base + BMOD,BMOD_DE);
-
- ret = synopmob_execute_command(base, cmd, arg);
- if ( !ret ) {
- ret = ERRIDMA;
- err = rt_sem_take(sdc->sem, DMA_TRANSFER_TIMEOUT_TICKS);
- if ( !err ) {
- while (--loop_for_command_done_check > 0) {
- intsts = synopmob_read_register(base+RINTSTS);
- if ((intsts & (INTMSK_CMD_DONE|INTMSK_DAT_OVER)) == (INTMSK_CMD_DONE|INTMSK_DAT_OVER)) {
- break;
- }
- plat_loop(1);
- }
- ret = synopmob_handle_standard_rinsts(intsts);
- if (!ret ) {
- if( !loop_for_command_done_check || !(sdc->idsts & 0x100)) { //normal interrupt
- ret = ERRIDMA;
- }
- }
- }
- }
- if (!ret) {
- synopmob_set_register(base + RINTSTS, 0xfffe); //clear interrupts
- synopmob_clear_bits(base + CTRL, CTRL_USE_IDMAC);
- synopmob_clear_bits(base + BMOD,BMOD_DE);
- synopmob_set_register(base+BLKSIZ, 512);
- synopmob_set_register(base+BYTCNT, 512);
- }
- else {
- char* op = "read";
- if (rw)
- op = "write";
-
- SDIO_PRINTF("sdio_drv_chain_write1(%s) fail:, ret = %d\n", op, ret);
- }
- if ( rw && !ret ) {
- ret = synopmob_wait_data_ready(base);
- }
- rt_sem_release(sdc->mutex);
- return ret;
- //return ret ? 0/*false*/ : 1/*true*/;
- }
- #endif
- #endif
- static int sdio_card_reset(sdc_t* sdc)
- {
- unsigned int resp;
- int ret;
- /* Soft Reset card */
- sdio_drv_creg_write(sdc, 0x6, 0, 0x8, &resp);
- return 0;
- }
- static int enum_sdio_card(sdc_t* sdc)
- {
- int ret;
- unsigned int resp;
- unsigned int base = sdc->ip_base;
- #if 0
- synopmob_set_bits(0x98500004, (1<<24)); //set to output mode
- synopmob_set_bits(0x98500000, (1<<24)); //power off
- plat_loop(1000000/5); //Lets give some ramp down period
- synopmob_clear_bits(0x98500000, (1<<24)); //power on
- plat_loop(1000000/5);//Lets give some ramp down period
- #endif
- synopmob_set_register(base+CTYPE, ONE_BIT_MODE);
- synopmob_set_register(base+CLKENA, 0x00000001); /*enable clock, non-low-power mode*/
- ret = synopmob_set_clk_freq(sdc, MMC_FOD_DIVIDER_VALUE);
-
- if ( !ret ) {
- plat_loop(100); //enough for 74 clock.
- #if 0
- sdio_card_reset(sdc);
- plat_loop(100000);
- #endif
- ret = sd_send_cmd5(sdc, 0, &resp);
- if (!ret) {
- resp &= 0x00ffffff;
- ret = sd_send_cmd5(sdc, resp, &resp);
- }
- }
- if (!ret) {
- ret = sd_send_cmd3(sdc); //get RCA
- }
-
- if (!ret) {
- ret = sd_send_cmd7(sdc); //select the card
- }
- if (!g_use_bcm43362)
- {
- sdio_drv_creg_read(sdc, 0x13, 0, &resp);
- if ((resp & 1) && (sdc->wkmod & (SDC_WKMOD_4WIRE|SDC_WKMOD_25M_STAND_SPEED|SDC_WKMOD_50M_HI_SPEED))){ //high speed support?
- if (sdc->wkmod & SDC_WKMOD_4WIRE) {
- sdio_drv_creg_read(sdc, 0x7, 0, &resp);
- resp &= 0xfc;
- resp |= (1 << 1);
- sdio_drv_creg_write(sdc, 0x7, 0, resp, &resp); //switch to 4bit mode
- sdio_drv_creg_read(sdc, 0x7, 0, &resp);
- if ((resp & 0x3) != 0x2) {
- return ERRCARDINTERNAL; // 4bit mode failed
- }
- synopmob_set_register(base+CTYPE, FOUR_BIT_MODE);
- }
- if (sdc->wkmod & (SDC_WKMOD_25M_STAND_SPEED|SDC_WKMOD_50M_HI_SPEED)) {
- ret = synopmob_set_clk_freq(sdc, ONE_BIT_BUS_FREQ);
- //ret = synopmob_set_clk_freq(sdc, 0);
- }
- }
- sdio_drv_creg_read(sdc, 0x3, 0, &resp);
- if (!ret) {
- sdio_drv_creg_read(sdc, 0x0, 0, &resp); //card version
- sdio_drv_creg_write(sdc, 0x4, 0, 0x3, &resp); //enable interrupts in card
- sdio_drv_creg_write(sdc, 0x2, 0, 0x2, &resp); //Eable IO in card
- do {
- sdio_drv_creg_read(sdc, 0x3, 0, &resp);
- } while (!(resp & 2));
- }
- } //g_use_bcm43362
- sdc->card_type = SDIO_TYPE;
- synopmob_set_bits(base+CTRL, INT_ENABLE);
-
- return ret;
- }
- int sdio_high_speed_mode(HSDC handle, int bitwidth, int freq)
- {
- int ret;
- sdc_t* sdc = (sdc_t*)handle;
- if (bitwidth == 4)
- {
- synopmob_set_register(sdc->ip_base+CTYPE, FOUR_BIT_MODE);
- }
- ret = synopmob_set_clk_freq(sdc, /*ONE_BIT_BUS_FREQ*/1);
- if (ret != 0)
- {
- SDIO_PRINTF("sdio_high_speed_mode fail:, ret = %d\n", ret);
- }
- return ret;
- }
- static int common_init(unsigned int which, unsigned int sdio, unsigned int wkmod, unsigned int* dma_desc, HSDC* phandle)
- {
- int ret = ERRNORES;
- sdc_t* sdc;
- unsigned int base;
- unsigned int temp;
- unsigned int fifo_thresh;
- volatile DmaDesc *pDmaDesc;
- rt_sem_t sem;
- rt_sem_t mutex;
-
- base = SDC0_REG_BASE;
- temp = PMU_SDC0_RST_BIT;
- if (which > 0) {
- base = SDC1_REG_BASE;
- temp = PMU_SDC1_RST_BIT;
- }
-
- #if 0
- //PMU_RST_MODULE(temp); plat_loop(1);
- temp = synopmob_read_register(PMU_REG_CLK_DIV3);
- temp &= (~(0x0f<<8));
- temp |= (0xf<<8);
- synopmob_set_register(PMU_REG_CLK_DIV3, temp);
- #endif
- *phandle = (HSDC)0;
-
- sdc = &sdc_array[which];
- sem = sdc->sem;
- mutex = sdc->mutex;
- memset((void *)sdc, 0, sizeof(*sdc));
- sdc->wkmod = wkmod;
- sdc->idma_support = 0;
- sdc->ip_base = base;
- sdc->rca = 0;
- sdc->card_type = NONE_TYPE;
- if (!sem) {
- sem = rt_sem_create("fh_sdio_sem", 0, RT_IPC_FLAG_PRIO);//OSSemCreate (0);
- if ( !sem ) {
- return ret;
- }
- }
- sdc->sem = sem;
- if (!mutex) {
- mutex = rt_sem_create("fh_sdio_mutex", 1, RT_IPC_FLAG_PRIO);//OSSemCreate (1);
- if ( !mutex ) {
- return ret;
- }
- }
- sdc->mutex = mutex;
-
- synopmob_set_bits(base + CTRL, CTRL_RESET); //reset host controller
- plat_loop(100);
- synopmob_clear_bits(base + CTRL,CTRL_USE_IDMAC);
- sdc->idma_support = 1; //fixed to support IDMA
- pDmaDesc = (volatile DmaDesc *)dma_desc;
- sdc->pDmaDesc = pDmaDesc;
- if (sdc->idma_support) {
- synopmob_set_bits(base + CTRL, DMA_RESET);
- plat_loop(100);
- synopmob_set_bits(base + CTRL, FIFO_RESET);
- plat_loop(100);
- synopmob_set_bits(base + BMOD, BMOD_SWR);
- plat_loop(100);
-
- //synopmob_set_bits(base + BMOD,BMOD_DE);
- pDmaDesc->desc0 = DescSecAddrChained;
- pDmaDesc->desc1 = 0;
- pDmaDesc->desc2 = 0;
- pDmaDesc->desc3 = (unsigned int)(pDmaDesc);
- synopmob_set_register(base + DBADDR, (unsigned int)(pDmaDesc));
- }
- synopmob_set_register(base+CTYPE, ONE_BIT_MODE);
- synopmob_set_register(base+RINTSTS, 0xffffffff);//clear interrupt.
- synopmob_clear_bits(base+CTRL, INT_ENABLE);
- synopmob_set_register(base+INTMSK, 0); // mask all INTR
- synopmob_set_register(base+IDINTEN, IDMAINTBITS); //Enable DMA INTR
- synopmob_set_register(base+TMOUT, 0xffffffff); /* Set Data and Response timeout to Maximum Value*/
- /* Set the card Debounce to allow the CDETECT fluctuations to settle down*/
- synopmob_set_register(base+DEBNCE, 0x0FFFFF);
- fifo_thresh = synopmob_read_register(base+FIFOTH);
- //fifo_thresh = GET_FIFO_DEPTH(fifo_thresh) / 2;
- fifo_thresh = (GET_FIFO_DEPTH(fifo_thresh) + 1) / 2;
- sdc->fifo_depth = fifo_thresh * 2;
- sdc->fifo_threth = fifo_thresh;
- /* Tx Watermark */
- synopmob_clear_bits(base+FIFOTH, 0xfff);
- synopmob_set_bits(base+FIFOTH, fifo_thresh);
- /* Rx Watermark */
- synopmob_clear_bits(base+FIFOTH, 0x0fff0000);
- synopmob_set_bits(base+FIFOTH, (fifo_thresh-1) << 16);
- //synopmob_set_bits(base+FIFOTH, 2<< 28);
- if (!sdio) {
- ret = enum_sd_card(sdc);
- }
- else {
- ret = enum_sdio_card(sdc);
- }
- if (!ret) {
- *phandle = (HSDC)sdc;
- }
- return ret;
- }
- int sdc_is_connected(unsigned int which)
- {
- unsigned int base = SDC0_REG_BASE;
-
- if (which > 0)
- base = SDC1_REG_BASE;
- return !(synopmob_read_register(base+CDETECT) & 1);
- }
- int sdc_init(unsigned int which, unsigned int wkmod, unsigned int* dma_desc, HSDC* phandle)
- {
- return common_init(which, 0, wkmod, dma_desc, phandle);
- }
- int sdio_init(unsigned int which, unsigned int wkmod, unsigned int* dma_desc, HSDC* phandle)
- {
- return common_init(which, 1, wkmod, dma_desc, phandle);
- }
- int sdio_enable_card_int(HSDC handle, int enable)
- {
- unsigned int base = ((sdc_t*)handle)->ip_base;
- if (enable) {
- //synopmob_set_register(base+INTMSK, INTMSK_SDIO);
- synopmob_set_register(base+INTMSK, (synopmob_read_register(base+INTMSK) | INTMSK_SDIO ));
- }
- else {
- //synopmob_set_register(base+INTMSK, 0);
- synopmob_set_register(base+INTMSK, (synopmob_read_register(base+INTMSK) & ~INTMSK_SDIO ));
- }
- return 0;
- }
- int sdio_set_card_int_cb(HSDC handle, void (*cb)(void))
- {
- ((sdc_t*)handle)->cb = cb;
- return 0;
- }
- static void OSSDCISR(sdc_t* sdc)
- {
- unsigned int sts;
- unsigned int base;
- base = sdc->ip_base;
- sts = synopmob_read_register(base+IDSTS);
- if ( sts ) {
- synopmob_set_register(base+IDSTS, sts);
- sdc->idsts = sts;
- rt_sem_release(sdc->sem);
- }
- //sts = synopmob_read_register(base+RINTSTS);
- sts = synopmob_read_register(base+MINTSTS);
- sts &= INTMSK_SDIO;
- if ( sts ) { //interrupt from WIFI card.
- //synopmob_set_register(base+INTMSK, 0); //mask all the interrupt
- synopmob_set_register(base+INTMSK, synopmob_read_register(base+INTMSK) & ~INTMSK_SDIO ); //mask sdio interrupt
- synopmob_set_register(base+RINTSTS, sts);
- synopmob_set_register(base+MINTSTS, sts);
- if (sdc->cb) {
- sdc->cb();
- }
- }
- }
- void OSSDCINTR_0(int vector, void *param)
- {
- OSSDCISR(&sdc_array[0]);
- }
- void OSSDCINTR_1(int vector, void *param)
- {
- OSSDCISR(&sdc_array[1]);
- }
- void fh_sdio0_init(void)
- {
- int sd0_irq = SDC0_IRQn;
-
- rt_hw_interrupt_install(sd0_irq, OSSDCINTR_0, NULL, NULL);
- rt_hw_interrupt_umask(sd0_irq);
- }
- void fh_sdio1_init(void)
- {
- int sd1_irq = SDC1_IRQn;
-
- rt_hw_interrupt_install(sd1_irq, OSSDCINTR_1, NULL, NULL);
- rt_hw_interrupt_umask(sd1_irq);
- }
- void fh_sdio_init(void)
- {
- fh_sdio0_init();
- fh_sdio1_init();
- }
- int sdc_deinit(HSDC handle)
- {
- return -1; // TBD... fix me
- }
- int sdc_set_clk_divider(unsigned int divider)
- {
- if(divider > 255)
- return -1;
-
- sdc_clk_divider = divider;
- return 0;
- }
|