久久久久久久999_99精品久久精品一区二区爱城_成人欧美一区二区三区在线播放_国产精品日本一区二区不卡视频_国产午夜视频_欧美精品在线观看免费

 找回密碼
 立即注冊

QQ登錄

只需一步,快速開始

搜索
查看: 7613|回復: 1
打印 上一主題 下一主題
收起左側

mp3(2)

[復制鏈接]
跳轉到指定樓層
樓主
ID:3271 發表于 2008-9-1 16:45 | 只看該作者 回帖獎勵 |倒序瀏覽 |閱讀模式

//*********************************************************
// MP3ファイルを淋す
// 苞眶¨
//  num この眶だけ斧つけたら提る
//  この眶笆布であってもディレクトリエントリを鏈て拇べ姜わったら提る
// 提り猛¨
//  斧つけた眶
//  DataBuff[0×31] 呵稿に斧つかった MP3のディレクトリ攫鼠
//*********************************************************
uint search_MP3(uint num){
 uint ent;
 ulong sec;
 uchar i, j;
 uint n;

 n = 0;
 sec = (ulong)DIRstart * 512;
 for (ent = 0; ent < RootDirEntriesCount; ){
  if (SPI_read_open(sec)) return n;
  for (i = 0; i < 16; i++){
   for (j = 0; j < 32; j++){
    DataBuff[j] = SPI_in();
   }
   if (DataBuff[0] == 0) break; // 踏蝗脫撾拌浮叫
   if (((DataBuff[26] != 0)||(DataBuff[27] != 0))&& // クラスタ戎規
    (DataBuff[0] != 0xE5)&&      // 猴近マ〖ク
    (!(DataBuff[11] & 0x18))&&     // 奶撅ファイル
    (DataBuff[8]=='M')&&(DataBuff[9]=='P')&&(DataBuff[10]=='3')){
               // 橙磨灰MP3
    n++;
    if (--num == 0) break;
   }
   ent++;
  }
  for (++i; i < 16; i++){
   for (j = 0; j < 32; j++){
    SPI_in(); // セクタ尸の荒りを粕み嘉て
   }
  }
  SPI_read_close();
  if (DataBuff[0] == 0) break; // 踏蝗脫撾拌浮叫
  if (num == 0) break;
  sec += 512; // 肌のセクタ
 }
 return n;
}


//****************************************
// MP3ファイルを浩欄
// fn:  妒戎規
// clst: 遍琳倡幌クラスタ。==0 の眷圭は呵介から
// remainsec: 浩欄デ〖タ荒りセクタ眶
//****************************************
char play_music(uint fn, uint clst, long remainsec){
 ulong sec;
 uint remain;
 uint cn;
 uint i;

 if (fn != search_MP3(fn)){
  return -1;

 }


 MusicSize = (ulong)DataBuff[28] + ((ulong)DataBuff[29]<<8)
     + ((ulong)DataBuff[30]<<16) + ((ulong)DataBuff[31]<<24);

 if (clst){
  // 龐面から浩欄
  Cluster = clst;
  RemainSec = remainsec;
  remain = MusicSize - (remainsec * 512);
 } else {
  // 呵介から浩欄
  Cluster = (uint)DataBuff[26] + ((uint)DataBuff[27] << 8);
  RemainSec = MusicSize / 512;
  remain = MusicSize % 512;
 }

 if ((Cluster == 0xFFFF)||(Cluster < 2)){
  return -2; // クラスタ佰撅
 }

 // 肌のクラスタを拇べておく
 NextCluster = next_cluster(Cluster, 0, 0);

 StopSw = 0;
 RewSw = 0;
 FfSw = 0;

 cn = SectorsPerCluster;
 sec = (((ulong)Cluster - 2) * (ulong)SectorsPerCluster + DataStart) * 512;

 MP3_init();
 Timer = 100; // READYタイマ〖セット

 while (RemainSec > 0){
  uchar data;
  char bit;

  if (SPI_read_open(sec)) return -2;
  cli();    // 充哈み敦賄
  SPI_out(0xFF);  // 1byte謄のクロック流叫 & WAIT

  for (i = 0; i < 512; i++){

   while (!(inp(PIND)&(1<<MP3_DREQ))){ // VS1001 BUSYチェック

    WDR;   // ウォッチドッグタイマ〖リセット
    Timer = 100; // READYタイマセット

    if (NextCluster == 0){
     // 肌のクラスタを拇べる
     NextCluster = next_cluster(Cluster, sec, i);
    } else {
     sei(); // 充り哈み釣材
     SLEEP;
     cli(); // 充哈み敦賄
    }
   }

   data = inp(SPDR);
   outp(0xFF, SPDR); // SD/MMCカ〖ドへ肌のデ〖タリクエスト

   if (data & 0x80){
    sbi(PORTD, MP3_SI);
   } else {
    cbi(PORTD, MP3_SI);
   }
   sbi(PORTD, MP3_BSYNC); // BSYNC = H
   sbi(PORTD, MP3_DCLK);
   data <<= 1;
   cbi(PORTD, MP3_DCLK);
   cbi(PORTD, MP3_BSYNC); // BSYNC = L

   for (bit = 0; bit < 7; bit++){
    if (data & 0x80){
     sbi(PORTD, MP3_SI);
    } else {
     cbi(PORTD, MP3_SI);
    }
    sbi(PORTD, MP3_DCLK);
    data <<= 1;
    cbi(PORTD, MP3_DCLK);
   }
  }
  sei(); // 充り哈み釣材
  SPI_read_close();

  --RemainSec;
 
  if (--cn > 0){
   sec += 512; // 肌のセクタ

   if (Debug && (cn == 1)){
    uint n;
    n = BattVolt;
    WriteHex(n>>8);
    WriteHex(n);
    WriteCom(' ');
    WriteHex(Cluster>>8);
    WriteHex(Cluster);
    WriteCom('\n');
   }
  } else {
   // 肌のクラスタ
   if (NextCluster == 0){
    Cluster = next_cluster(Cluster, 0, 0);
   } else {
    Cluster = NextCluster;
    NextCluster = 0;
   }

   if ((Cluster == 0xFFFF)||(Cluster < 2)){
    // 佰撅クラスタが浮叫されたので動擴姜位
    remain = 0;
    break;
   }

   sec = (((ulong)Cluster - 2) * (ulong)SectorsPerCluster
       + DataStart) * 512;
   cn = SectorsPerCluster;

  }

  check_batt(); // バッテリ〖チェック

  if (Timer == 0){ // READYタイムアウトしたら
   if (NoBusyRetry > 0){ // リトライしてもダメなら
    beep(1, BEEP_LOW); // 你不ビ〖プ1攙
    NoBusyRetry = 0;
    FfSw = 0;
    return 2;  // FF胺いで姜位 ⅹ 肌の妒へ
   } else {
    NoBusyRetry = 1;
    RewSw = 0;
    return 3;  // REW胺いで姜位 ⅹ その妒の呵介からリトライ
   }
  }

  // STOP, FFボタンチェック
  if (StopSw > 2){
   beep(1, BEEP_HIGH);
   NoBusyRetry = 0;
   return 1;
  } else if (FfSw > 2){
   beep(1, BEEP_HIGH);
   NoBusyRetry = 0;
   return 2;
  } else if (RewSw > 2){
   beep(1, BEEP_HIGH);
   NoBusyRetry = 0;
   return 3;
  }
 }

 NoBusyRetry = 0;

 // 1セクタ(512bytes)踏塔の尸を浩欄
 if ((remain > 0)&&(remain < 512)){
  uchar data;
  char bit;

  if (SPI_read_open(sec)) return -2;
  cli();    // 充哈み敦賄
  SPI_out(0xFF);  // 1byte謄のクロック流叫□WAIT

  for (i = 0; i < 512; i++){
   data = inp(SPDR);
   outp(0xFF, SPDR);
   if (remain-- > 0){
    while (!(inp(PIND)&(1<<MP3_DREQ))){
     // VS1001 BUSY
     WDR;  // ウォッチドッグタイマクリア
     sei();   // 充り哈み釣材
     SLEEP;
     cli();   // 充哈み敦賄
    }
    sbi(PORTD, MP3_BSYNC); // BSYNC = H
    for (bit=0; bit<8; bit++){
     if (data & 0x80){
      sbi(PORTD, MP3_SI);
     } else {
      cbi(PORTD, MP3_SI);
     }
     sbi(PORTD, MP3_DCLK);
     data <<= 1;
     cbi(PORTD, MP3_DCLK);
     cbi(PORTD, MP3_BSYNC); // BSYNC = L
    }
   }
  }

  sei(); // 充り哈み釣材
  SPI_read_close();
 }

 return 0;
}


//****************************************
// 肌のクラスタを滇める
//****************************************
uint next_cluster(uint c, ulong sec, uint remain){
 ulong addr;
 uint data;
 ulong fatadr;
 uint i;

 if (FATtype == 0){
  // FAT12
  addr = (ulong)FATstart * 512 + (ulong)c + (c >> 1);
  if ((addr & 511) == 511){
   if (sec != 0){
    // 粕み哈みオ〖プン面なら粕み嘉て
    for (i = remain; i < 512; i++) SPI_in();
    SPI_read_close();

    data = read_word(addr) & 0x00FF;
    data |= ((read_word(addr+1) & 0x00FF) << 8);

    // 浩刨オ〖プン、笆漣の疤彌まで粕み嘉て
    if (SPI_read_open(sec)) return 0;
    SPI_in();
    for (i = 0; i < remain; i++) SPI_in();
   } else {
    // セクタをまたぐ
    data = read_word(addr) & 0x00FF;
    data |= ((read_word(addr+1) & 0x00FF) << 8);
   }
  } else {
   fatadr = addr & 0xFFFFFE00;
   if ((sec != 0)&&(fatadr != FatCacheAddr)){
    // 粕み哈みオ〖プン面で FATバッファ痰跟なら粕み嘉て
    for (i = remain; i < 512; i++) SPI_in();
    SPI_read_close();

    data = read_word(addr);

    // 浩刨オ〖プン、笆漣の疤彌まで粕み嘉て
    if (SPI_read_open(sec)) return 0xFFFF;
    SPI_in();
    for (i = 0; i < remain; i++) SPI_in();
   } else {
    data = read_word(addr);
   }
  }
  if (c & 1){
   c = data >> 4;
  } else {
   c = data & 0x0fff;
  }
  if (c >= 0x0ff8) c = 0xFFFF;
 } else {
  // FAT16
  addr = (ulong)FATstart * 512 + (ulong)c * 2;
  fatadr = addr & 0xFFFFFE00;
  if ((sec != 0)&&(fatadr != FatCacheAddr)){
   // 粕み哈みオ〖プン面で FATバッファ痰跟なら粕み嘉て
   for (i = remain; i < 512; i++) SPI_in();
   SPI_read_close();

   c = read_word(addr);

   // 浩刨オ〖プン、笆漣の疤彌まで粕み嘉て
   if (SPI_read_open(sec)) return 0;
   SPI_in();
   for (i = 0; i < remain; i++) SPI_in();
  } else {
   c = read_word(addr);
  }
  if (c >= 0xFFF8) c = 0xFFFF;
 }

 return c;
}


//****************************************
// MMC/SDカ〖ドから 2byteを粕む
// セクタ董腸のアクセスはできないので
// 懼疤供鎳で雇胃すること
//****************************************
uint read_word(ulong addr){
 uint w;
 uint b;
 uchar *p;

 b = addr & 511;
 addr &= 0xFFFFFE00;

 if (addr != FatCacheAddr){
  if (SPI_read_open(addr)) return 0;
  for (p = FatCache; p < (FatCache + 512); ){
   *p++ = SPI_in();
  }
  SPI_read_close();
  FatCacheAddr = addr;
 }

 w = (uint)FatCache[b++];
 w |= ((uint)FatCache << 8);

 return w;
}


//*******************************************************
// バッテリ〖排暗を拇べ、你すぎればシャットダウンする
//*******************************************************
int check_batt(void){

 if (BattVolt < BATT_SHUTDOWN){
  sbi(PORTB, SPI_CS);  // SPI CS=H
  beep(4, BEEP_LOW);
  cli(); // 充哈み敦賄
  if (RunningTimeH){
   write_eeprom(EEPROM_RUNNING_TIME_OLD_M, RunningTimeM);
   write_eeprom(EEPROM_RUNNING_TIME_OLD_H, RunningTimeH);
   RunningTimeM = 0;
   RunningTimeH = 0;
  }

  shutdown(0); // シャットダウン
 }

 return BattVolt;
}


//*********************************************************
// シャットダウン
// flag != 0 の眷圭は EEPROM今き哈みを乖わずに排富を磊る
//*********************************************************
void shutdown(char flag){

 cli(); // 充哈み敦賄
 sbi(PORTB, SPI_CS);  // SPI CS=H
 cbi(PORTB, MUTE);  // 不蘭ミュ〖ト
 MP3_command(0, 0x0010); // VS1001k powerdown
 if (!flag){
  write_eeprom(EEPROM_VOLUME, Volume);
  write_eeprom(EEPROM_MUSIC, PlayMusic);
  write_eeprom(EEPROM_CLUSTER_L, Cluster);
  write_eeprom(EEPROM_CLUSTER_H, Cluster>>8);
  write_eeprom(EEPROM_REMAIN1, RemainSec);
  write_eeprom(EEPROM_REMAIN2, RemainSec>>8);
  write_eeprom(EEPROM_REMAIN3, RemainSec>>16);
  write_eeprom(EEPROM_SIZE1, MusicSize);
  write_eeprom(EEPROM_SIZE2, MusicSize>>8);
  write_eeprom(EEPROM_SIZE3, MusicSize>>16);
  write_eeprom(EEPROM_SIZE4, MusicSize>>24);
  write_eeprom(EEPROM_RUNNING_TIME_M, RunningTimeM);
  write_eeprom(EEPROM_RUNNING_TIME_H, RunningTimeH);
 }
 for(;;){    // 痰嘎ル〖プ
  cbi(PORTC, POWER); // Power down
  WDR;
  SLEEP;
 }
}


//*******************************************************
// EEPROM 今き哈み
//*******************************************************
void write_eeprom(uchar adrs, uchar data){

 // 票じ柒推なら今き哈み瓢侯を乖わない
 if (read_eeprom(adrs) != data){
  while(inp(EECR) & (1<<EEWE));
  outp(adrs >> 8, EEARH);
  outp(adrs, EEARL);
  outp(data, EEDR);
  sbi(EECR, EEMWE);
  sbi(EECR, EEWE);
 }
}


//*******************************************************
// EEPROM 粕み哈み
//*******************************************************
uchar read_eeprom(uchar adrs){

 while(inp(EECR) & (1<<EEWE));
 outp(adrs >> 8, EEARH);
 outp(adrs, EEARL);
 sbi(EECR, EERE);
 return inp(EEDR);
}


//****************************************
// BEEP
// 苞眶¨ n 棠瓢攙眶
//   err != 0 ならエラ〖不(你い不)
// 充り哈み釣材にすることに廟罷
//****************************************
void beep(uchar n, char err){
 char i;

 cli(); // MP3_init()借妄面に不翁拇淚されたら氦るので充哈み敦賄
 MP3_init();
 sei(); // 充り哈み釣材
 for (; n > 0; --n){
  if (err){
   MP3_sin(ErrorBeep);
  } else {
   MP3_sin(BeepData);
  }

  // 箕粗略ち。この粗が BEEPの不の墓さになる
  for (i=0; i < 10*2; i++){ // 腆100ms
   SLEEP;
  }

  MP3_sin(BeepStop);
 
  // BEEPの不粗
  for (i=0; i < 3*2; i++){ // 腆30ms
   SLEEP;
  }
 }

 MP3_sin(BeepStop); // 箕」賄まらないことがあるのでもう辦攙
}


//****************************************
// EasyMP3 介袋步
//****************************************
void MP3_init(void){

 MP3_command(0, 0x0004);   // Soft Reset
 InitWait();
 MP3_command(0, 0x0000);
 while(!(inp(PIND)&(1<<MP3_DREQ)));

 MP3_command(11, ((uint)Volume << 8)|((uint)Volume)); // 不翁肋年
 MP3_command(3, 0x8000 + (uint)((VS1001_CLOCK)/2000));  // Clock
 InitWait();
}


//****************************************
// EasyMP3 テスト脫賴腹僑叫蝸
//****************************************
void MP3_sin(PGM_VOID_P data){
 uchar i;
 uchar bit;
 uchar c;

 for (i=0; i < 8; i++){
  c = (uchar)PRG_RDB(data++);
  sbi(PORTD, MP3_BSYNC);
  for (bit=0; bit<8; bit++){
   if (c & 0x80){
    sbi(PORTD, MP3_SI);
   } else {
    cbi(PORTD, MP3_SI);
   }
   sbi(PORTD, MP3_DCLK);
   c <<= 1;
   cbi(PORTD, MP3_DCLK);
   cbi(PORTD, MP3_BSYNC);
  }
  sbi(PORTD, MP3_SI);
 }
}


//****************************************
// EasyMP3 コマンド叫蝸
//****************************************
void MP3_command(uchar addr, uint arg){

 cbi(PORTD, MP3_CS);
 MP3_com_write(2);
 MP3_com_write(addr);
 MP3_com_write(arg >> 8);
 MP3_com_write(arg);
 sbi(PORTD, MP3_CS);
}


//****************************************
// EasyMP3 コマンド掐蝸 踏蝗脫
//****************************************
/*
uint MP3_command_read(uchar addr){
 char i;
 uint r = 0;

 cbi(PORTD, MP3_CS);
 MP3_com_write(3);
 MP3_com_write(addr);
 for(i=0; i<16; i++){
  sbi(PORTD, MP3_SCLK);
  r <<= 1;
  if (inp(PINB) & (1<<MP3_SO)){
   r |= 1;
  }
  cbi(PORTD, MP3_SCLK);
 }
 sbi(PORTD, MP3_CS);
 return r;
}
*/


//****************************************
// EasyMP3 コマンド 1byte叫蝸
//****************************************
void MP3_com_write(uchar data){
 char i;

 for (i=0; i<8; i++){
  if (data & 0x80){
   sbi(PORTD, MP3_SI);
  } else {
   cbi(PORTD, MP3_SI);
  }
  sbi(PORTD, MP3_SCLK);
  data <<= 1;
  cbi(PORTD, MP3_SCLK);
 }
 sbi(PORTD, MP3_SI);
}


//****************************************
// MMC/SDカ〖ドから 32byteを粕んで DataBuffに呈羌
//****************************************
void read_32(ulong addr){
 uint  i, b;

 if (SPI_read_open(addr & 0xFFFFFE00)) return;
 b = (uint)addr & 0x01FF;
 for (i=0; i < 512; i++){
  if (i == b){
   int n;
   for (n = 0; n < 32; n++){
    DataBuff[n] = SPI_in();
    i++;
   }
  } else {
   SPI_in();
  }
 }
 SPI_read_close();
}


//****************************************
// 介袋步箕のクロックウエイト
//****************************************
void InitWait(void){
 uchar i;

 for(i=0; i< 255; i++){
  NOP;
 }
}


//****************************************
// MMC/SDカ〖ド粕み哈みオ〖プン
//****************************************
uchar SPI_read_open(ulong adrs){
 uint i;
 uchar r;
 uchar retry;

 // カ〖ドによってはリトライが澀妥のようだ
 for (retry = 0; retry < 100; retry++){
  cbi(PORTB, SPI_CS);  // SPI CS=L
  r = SPI_command(17, adrs);
  if (r == 0){
   for (i = 0; i < 2000; i++){ // ものすごく略つ眷圭がある
    r = SPI_in();
    if (r == 0xFE){
     return 0; // 啪流倡幌マ〖ク
    }
   }
  } else if (r == 0xFE){
   return 0; // 賴撅オ〖プン
  }
  sbi(PORTB, SPI_CS);  // SPI CS=H
  if (Debug){
   WriteHex(r);
   WriteCom(' ');
  }
//  InitWait();
 }

 return 1; // オ〖プンできなかった
}


//****************************************
// MMC/SDカ〖ド粕み哈みクロ〖ズ
// 1byteのずれがあっても丹にしない
//****************************************
void SPI_read_close(void){
 uint i;

 SPI_in(); // CRC
 SPI_in(); // CRC
 // BUSYチェック
 for (i = 0; i < 1000; i++){
  if (SPI_in() == 0xFF) break;
 }
 sbi(PORTB, SPI_CS);  // SPI CS=H
}


//****************************************
// MMC/SD SPIコマンド叫蝸
//****************************************
uchar SPI_command(uchar com, ulong arg){
 uint i;
 uchar r;

 // BUSYチェック 絲家のカ〖ドでは澀寇みたい
 for (i = 0; i < 1000; i++){
  r = SPI_in();
  if (r == 0xFF) break;
 }
 if (r != 0xFF) return 1;

 SPI_out(com|0x40);
 SPI_out(arg >> 24);
 SPI_out(arg >> 16);
 SPI_out(arg >> 8);
 SPI_out(arg);
 SPI_out(0x95);  // 介袋步コマンドのCRC

 for(i = 0; i < 1000; i++){
  r = SPI_in();
  if (r == 0xFE)  break;
  if (!(r & 0x80)) break;
 }
 return r;
}


//****************************************
// MMC/SD SPIデ〖タ叫蝸
//****************************************
void SPI_out(uchar data){

 outp(data, SPDR);
 while(!(inp(SPSR) & 0x80));
}


//****************************************
// MMC/SD SPIデ〖タ掐蝸
//****************************************
uchar SPI_in(void){

 outp(0xFF, SPDR);
 while(!(inp(SPSR) & 0x80));
 return inp(SPDR);
}


//****************************************
// MMC/SD SPIクロック姜位略ち
// SPDRをアクセスしてない眷圭は
// SPIF=0になって痰嘎ル〖プする眷圭があることに廟罷
//****************************************
void SPI_wait(void){
 while(!(inp(SPSR) & 0x80));
}


//****************************************
// 1セクタ〖粕み哈んで山績する
//****************************************
void print_sector(ulong sec){
 uchar r;
 int  i;

 sec *= 512;
 r = SPI_read_open(sec);
 WriteHex(r);
 WriteCom('\n');
 for (i=0; i<512; i++){
  r = SPI_in();
  WriteHex(r);
  if ((i & 31) == 31){
   WriteCom('\n');
  } else if (i & 1){
   WriteCom(' ');
  }
 }
 SPI_read_close();
}


 

分享到:  QQ好友和群QQ好友和群 QQ空間QQ空間 騰訊微博騰訊微博 騰訊朋友騰訊朋友
收藏收藏 分享淘帖 頂 踩
回復

使用道具 舉報

沙發
ID:15556 發表于 2009-8-8 12:53 | 只看該作者
朋友 本人想做搞mp3作為畢業設計 但看不明白耶
回復

使用道具 舉報

您需要登錄后才可以回帖 登錄 | 立即注冊

本版積分規則

手機版|小黑屋|51黑電子論壇 |51黑電子論壇6群 QQ 管理員QQ:125739409;技術交流QQ群281945664

Powered by 單片機教程網

快速回復 返回頂部 返回列表
主站蜘蛛池模板: 欧洲在线视频 | 亚洲一区二区日韩 | 国产成人在线播放 | 久久久久国产视频 | 国产超碰人人爽人人做人人爱 | www.玖玖玖| 成人国产精品久久 | 欧美亚洲视频在线观看 | 久久9久 | 91精品国产一区二区三区动漫 | 成人欧美一区二区三区在线播放 | 久久国产精品免费一区二区三区 | 亚洲精品乱码久久久久久蜜桃91 | 色久在线 | 一区二区不卡 | 91精品中文字幕一区二区三区 | 国产精品久久久久久久久久免费看 | 亚洲精品久久久 | 成人不卡视频 | 日操操| 成人免费视频久久 | 欧美一区免费 | 精品欧美一区二区三区 | 国产精品国产三级国产aⅴ中文 | 亚洲欧美在线观看 | 亚洲色片网站 | 国产精品九九九 | 久草资源在线 | 欧美国产视频 | 在线观看成人 | av在线伊人 | 欧美成人一区二免费视频软件 | 国产精品久久久久久一区二区三区 | 91成人在线 | 亚洲精品 在线播放 | 中文字幕亚洲视频 | 欧美精品乱码久久久久久按摩 | 精品久久久久一区二区国产 | 91麻豆蜜桃一区二区三区 | 日韩午夜网站 | 丁香久久|