////////////////////////////////////////////////////////////////////////////////////////// // // Licensetype: GNU General Public License (GPL) // // This program is free software: you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by // the Free Software Foundation, version 2.0. // // 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 2.0 for more details. // // A copy of the GPL 2.0 should have been included with the program. // If not, see http://www.gnu.org/licenses/ // // Official SVN repository and contact information can be found at // http://code.google.com/p/dolphin-emu/ // ////////////////////////////////////////////////////////////////////////////////////////// // Includes #include #include #include // so that we can test std::string == abc #include // for the pow() function #ifdef _WIN32 #include #endif #include "../Debugger/Debugger.h" #include "../Debugger/PBView.h" #include "Console.h" // open and close console, clear console window #include "../Globals.h" #include "../UCodes/UCodes.h" #include "../UCodes/UCode_AXStructs.h" #include "../UCodes/UCode_AX.h" #include "../UCodes/UCode_AXWii.h" // Externals u32 gLastBlock; extern int nFiles; float ratioFactor; // a global to get the ratio factor from MixAdd extern CDebugger* m_frame; //int PBSize = 128; // Parameter blocks std::vector gloopPos(NUMBER_OF_PBS); std::vector gsampleEnd(NUMBER_OF_PBS); std::vector gsamplePos(NUMBER_OF_PBS); // main std::vector gsrc_type(NUMBER_OF_PBS); std::vector gis_stream(NUMBER_OF_PBS); // PBSampleRateConverter src std::vector gratio(NUMBER_OF_PBS); std::vector gratiohi(NUMBER_OF_PBS); std::vector gratiolo(NUMBER_OF_PBS); std::vector gfrac(NUMBER_OF_PBS); std::vector gcoef(NUMBER_OF_PBS); // PBSampleRateConverter mixer std::vector gvolume_left(NUMBER_OF_PBS); std::vector gmix_unknown(NUMBER_OF_PBS); std::vector gvolume_right(NUMBER_OF_PBS); std::vector gmix_unknown2(NUMBER_OF_PBS); std::vector gmixer_control(NUMBER_OF_PBS); std::vector gmixer_vol1(NUMBER_OF_PBS); std::vector gmixer_vol2(NUMBER_OF_PBS); std::vector gmixer_vol3(NUMBER_OF_PBS); std::vector gmixer_vol4(NUMBER_OF_PBS); std::vector gmixer_vol5(NUMBER_OF_PBS); std::vector gmixer_vol6(NUMBER_OF_PBS); std::vector gmixer_vol7(NUMBER_OF_PBS); std::vector gmixer_d1(NUMBER_OF_PBS); std::vector gmixer_d2(NUMBER_OF_PBS); std::vector gmixer_d3(NUMBER_OF_PBS); std::vector gmixer_d4(NUMBER_OF_PBS); std::vector gmixer_d5(NUMBER_OF_PBS); std::vector gmixer_d6(NUMBER_OF_PBS); std::vector gmixer_d7(NUMBER_OF_PBS); // PBVolumeEnvelope vol_env std::vector gcur_volume(NUMBER_OF_PBS); std::vector gcur_volume_delta(NUMBER_OF_PBS); // PBAudioAddr audio_addr (incl looping) std::vector gaudioFormat(NUMBER_OF_PBS); std::vector glooping(NUMBER_OF_PBS); std::vector gloop1(NUMBER_OF_PBS); std::vector gloop2(NUMBER_OF_PBS); std::vector gloop3(NUMBER_OF_PBS); // PBADPCMInfo adpcm std::vector gadloop1(NUMBER_OF_PBS); std::vector gadloop2(NUMBER_OF_PBS); std::vector gadloop3(NUMBER_OF_PBS); // updates std::vector gupdates1(NUMBER_OF_PBS); std::vector gupdates2(NUMBER_OF_PBS); std::vector gupdates3(NUMBER_OF_PBS); std::vector gupdates4(NUMBER_OF_PBS); std::vector gupdates5(NUMBER_OF_PBS); std::vector gupdates_addr(NUMBER_OF_PBS); std::vector gupdates_data(NUMBER_OF_PBS); // Counters int j = 0; int k = 0; unsigned int l = 0; int iupd = 0; bool iupdonce = false; std::vector viupd(15); // the length of the update frequency bar int vectorLengthGUI = 8; // length of playback history bar for the GUI version int vectorLength = 15; // for console version int vectorLength2 = 100; // for console version // More stuff // should we worry about the additonal memory these lists require? bool will allocate // very little memory std::vector< std::vector > vector1(NUMBER_OF_PBS, std::vector(vectorLength, 0)); std::vector< std::vector > vector2(NUMBER_OF_PBS, std::vector(vectorLength2, 0)); std::vector numberRunning(NUMBER_OF_PBS); // Classes extern CDebugger* m_frame; // ======================================================================================= // Write title // -------------- std::string writeTitle(int a) { std::string b; if(a == 0) { b = " adpcm adpcm_loop\n"; b = b + " Nr pos / end lpos | voll volr | isl iss | pre yn1 yn2 pre yn1 yn2 | frac ratio[hi lo]\n"; } else if(a == 1) { b = " Nr pos / end lpos | voll volr | src form coef | 1 2 3 4 5\n"; } else if(a == 2) { b = " Nr pos / end lpos | voll volr | isl iss | e-l e-s\n"; } else if(a == 3) { b = " Nr voll volr dl dr curv delt mixc r | v1 v2 v3 v4 v5 v6 v7 | d1 d2 d3 d4 d5 d6 d7\n"; } return b; } // ======================================================================================= // ======================================================================================= // Write main message (presets) // -------------- std::string writeMessage(int a, int i) { char buf [1000] = ""; std::string sbuf; // ======================================================================================= // PRESETS // --------------------------------------------------------------------------------------- /* PRESET 0 " Nr pos / end lpos | voll volr | isl iss | pre yn1 yn2 pre yn1 yn2 | frac ratio[hi lo]\n"; "---------------|00 12341234/12341234 12341234 | 00000 00000 | 0 0 | 000 00000 00000 000 00000 00000 | 00000 00000[0 00000] PRESET 1 (updates) " Nr pos / end lpos | voll volr | src form coef | 1 2 3 4 5\n"; "---------------|00 12341234/12341234 12341234 | 00000 00000 | 0 0 0 | 0 0 0 0 0 PRESET 2 " Nr pos / end lpos | voll volr | isl iss | e-l e-s\n"; "---------------|00 12341234/12341234 12341234 | 00000 00000 | 0 0 | 000000 000000 */ if(a == 0) { sprintf(buf,"%c%02i %08i/%08i %08i | %05i %05i | %i %i | %03i %05i %05i %03i %05i %05i | %05i %05i[%i %05i]", 223, i, gsamplePos[i], gsampleEnd[i], gloopPos[i], gvolume_left[i], gvolume_right[i], glooping[i], gis_stream[i], gadloop1[i], gadloop2[i], gadloop3[i], gloop1[i], gloop2[i], gloop3[i], gfrac[i], gratio[i], gratiohi[i], gratiolo[i] ); } else if(a == 1) { sprintf(buf,"%c%02i %08i/%08i %08i | %05i %05i | %i %i %i | %i %i %i %i %i %08x %08x", 223, i, gsamplePos[i], gsampleEnd[i], gloopPos[i], gvolume_left[i], gvolume_right[i], gsrc_type[i], gaudioFormat[i], gcoef[i], gupdates1[i], gupdates2[i], gupdates3[i], gupdates4[i], gupdates5[i], gupdates_addr[i], gupdates_data[i] ); } else if(a == 2) { sprintf(buf,"%c%02i %08i/%08i %08i | %05i %05i | %i %i | %06i %06i", 223, i, gsamplePos[i], gsampleEnd[i], gloopPos[i], gvolume_left[i], gvolume_right[i], glooping[i], gis_stream[i], gsampleEnd[i] - gloopPos[i], gsampleEnd[i] - gsamplePos[i] ); } /* PRESET 3 " Nr voll volr dl dr curv delt mixc r | v1 v2 v3 v4 v5 v6 v7 | d1 d2 d3 d4 d5 d6 d7\n"; "---------------|00 00000 00000 00000 00000 00000 00000 00000 0 | 00000 00000 00000 00000 00000 00000 00000 | 00000 00000 00000 00000 00000 00000 00000 */ else if(a == 3) { sprintf(buf,"%c%02i %05i %05i %05i %05i %05i %05i %05i %i | %05i %05i %05i %05i %05i %05i %05i | %05i %05i %05i %05i %05i %05i %05i", 223, i, gvolume_left[i], gvolume_right[i], gmix_unknown[i], gmix_unknown2[i], gcur_volume[i], gcur_volume_delta[i], gmixer_control[i], (gmixer_control[i] & MIXCONTROL_RAMPING), gmixer_vol1[i], gmixer_vol2[i], gmixer_vol3[i], gmixer_vol4[i], gmixer_vol5[i], gmixer_vol6[i], gmixer_vol7[i], gmixer_d1[i], gmixer_d2[i], gmixer_d3[i], gmixer_d4[i], gmixer_d5[i], gmixer_d6[i], gmixer_d7[i] ); } sbuf = buf; return sbuf; } // ================ // ======================================================================================= // Collect parameters from Wii or GC // -------------- /* std::string ShowAllPB(int a, int i) { if(a == 0) { } else if(a == 1) { } else if(a == 1) { } else if(a == 1) } */ // ================ // ======================================================================================= // Collect parameters from Wii or GC // -------------- //inline void MixAddVoice(ParamBlockType &pb //void CollectPB(bool Wii, int i, AXParamBlockWii * PBw, ParamBlockType &pb) template void CollectPB(bool Wii, int i, ParamBlockType &PBs) //void CollectPB(bool Wii, int i, AXParamBlockWii * PBw, AXParamBlock * PBs) { // AXPB base gcoef[i] = PBs[i].coef_select; gloopPos[i] = (PBs[i].audio_addr.loop_addr_hi << 16) | PBs[i].audio_addr.loop_addr_lo; gsampleEnd[i] = (PBs[i].audio_addr.end_addr_hi << 16) | PBs[i].audio_addr.end_addr_lo; gsamplePos[i] = (PBs[i].audio_addr.cur_addr_hi << 16) | PBs[i].audio_addr.cur_addr_lo; // mixer (some differences) gvolume_left[i] = PBs[i].mixer.volume_left; gvolume_right[i] = PBs[i].mixer.volume_right; if(i == 94 && (PBs[i].mixer.unknown > 0 || PBs[i].mixer.unknown2 > 0)) DebugLog("(%i) | LOG Read Left: %04x | Right: %04x", i, PBs[i].mixer.unknown, PBs[i].mixer.unknown2); gmix_unknown[i] = PBs[i].mixer.unknown; gmix_unknown2[i] = PBs[i].mixer.unknown2; gmixer_control[i] = PBs[i].mixer_control; gcur_volume[i] = PBs[i].vol_env.cur_volume; gcur_volume_delta[i] = PBs[i].vol_env.cur_volume_delta; gmixer_vol1[i] = PBs[i].mixer.unknown3[0]; gmixer_vol2[i] = PBs[i].mixer.unknown3[2]; gmixer_vol3[i] = PBs[i].mixer.unknown3[4]; gmixer_vol4[i] = PBs[i].mixer.unknown3[6]; gmixer_vol5[i] = PBs[i].mixer.unknown3[0]; gmixer_vol6[i] = PBs[i].mixer.unknown3[2]; gmixer_vol7[i] = PBs[i].mixer.unknown3[4]; //gmixer_d1[i] = PBs[i].mixer.unknown4[1]; gmixer_d1[i] = PBs[i].next_pb_lo; gmixer_d2[i] = PBs[i].next_pb_hi; //gmixer_d2[i] = PBs[i].mixer.unknown4[3]; gmixer_d3[i] = PBs[i].mixer.unknown4[5]; gmixer_d4[i] = PBs[i].mixer.unknown4[7]; gmixer_d5[i] = PBs[i].mixer.unknown4[1]; gmixer_d6[i] = PBs[i].mixer.unknown4[3]; gmixer_d7[i] = PBs[i].mixer.unknown4[5]; // adpcm_loop_info (same in GC and Wii) gadloop1[i] = PBs[i].adpcm.pred_scale; gadloop2[i] = PBs[i].adpcm.yn1; gadloop3[i] = PBs[i].adpcm.yn2; gloop1[i] = PBs[i].adpcm_loop_info.pred_scale; gloop2[i] = PBs[i].adpcm_loop_info.yn1; gloop3[i] = PBs[i].adpcm_loop_info.yn2; // updates (differences) gupdates1[i] = PBs[i].updates.num_updates[0]; gupdates2[i] = PBs[i].updates.num_updates[1]; gupdates3[i] = PBs[i].updates.num_updates[2]; gupdates4[i] = PBs[i].updates.num_updates[3]; gupdates5[i] = PBs[i].updates.num_updates[4]; gupdates_addr[i] = (PBs[i].updates.data_hi << 16) | PBs[i].updates.data_lo; gupdates_data[i] = Memory_Read_U32(gupdates_addr[i]); gaudioFormat[i] = PBs[i].audio_addr.sample_format; glooping[i] = PBs[i].audio_addr.looping; gsrc_type[i] = PBs[i].src_type; gis_stream[i] = PBs[i].is_stream; // PBSampleRateConverter src gratio[i] = (u32)(((PBs[i].src.ratio_hi << 16) + PBs[i].src.ratio_lo) * ratioFactor); gratiohi[i] = PBs[i].src.ratio_hi; gratiolo[i] = PBs[i].src.ratio_lo; gfrac[i] = PBs[i].src.cur_addr_frac; } // =============== // ======================================================================================= // Prepare the condition that makes us show a certain block // -------------- template bool PrepareConditions(bool Wii, int i, ParamBlockType &PBs) { bool Conditions; if (m_frame->gOnlyLooping) // show only looping blocks { Conditions = PBs[i].audio_addr.looping; } else if (m_frame->gShowAll) // show all blocks { Conditions = true; } else if (m_frame->giShowAll > -1) // show all blocks { if (m_frame->giShowAll == 0) Conditions = (i < 31); else if(m_frame->giShowAll == 1) Conditions = (i > 30 && i < 61); else if(m_frame->giShowAll == 2) Conditions = (i > 60 && i < 91); else if(m_frame->giShowAll == 3) Conditions = (i > 90 && i < 121); } else // show only the ones that have recently been running { Conditions = (numberRunning.at(i) > 0 || PBs[i].audio_addr.looping); } return Conditions; } // =============== // I placed this in CUCode_AX because it needs access to private members of that class. //template //void CUCode_AX::Logging(short* _pBuffer, int _iSize, int a, bool Wii, AXParamBlockWii *PBs, int numberOfPBs) void CUCode_AX::Logging(short* _pBuffer, int _iSize, int a, bool Wii) { // Declare structures /**/ AXParamBlock PBs[64]; AXParamBlockWii PBw[NUMBER_OF_PBS]; int numberOfPBsWii = ReadOutPBsWii(m_addressPBs, PBw, NUMBER_OF_PBS - 1); int numberOfPBsGC = ReadOutPBs(m_addressPBs, PBs, 64); /**/ // Read out the number of PBs that have data int numberOfPBs; if(Wii) numberOfPBs = numberOfPBsWii; else numberOfPBs = numberOfPBsGC; /* DebugLog("After Read LOG: Left: %04x | Right: %04x || Left: %04x | Right: %04x ", PBs[94].mixer.unknown, PBs[94].mixer.unknown2, PBw[94].mixer.unknown, PBw[94].mixer.unknown2 ); */ // Select blocks to show bool Conditions; // ======================================================================================= // Update parameter values // -------------- // We could chose to update these only if a block is currently running. Later I'll add options // to see both the current and the latest active value. //if (PBs[i].running) int irun = 0; for (int i = 0; i < numberOfPBs; i++) { // -------------------------------------------------------------------- // Write a line for the text log if nothing is playing // -------------- if (PBs[i].running) { irun++; } if (i == numberOfPBs - 1 && irun == 0) { for (int i = 0; i < nFiles; i++) { std::string sfbuff; sfbuff = "-----"; aprintf(i, (char *)sfbuff.c_str()); } } // -------------- // Prepare conditions /**/ if(Wii) Conditions = PrepareConditions(Wii, i, PBw); else Conditions = PrepareConditions(Wii, i, PBs); if (Conditions) { // Collect parameters if(Wii) CollectPB(Wii, i, PBw); else CollectPB(Wii, i, PBs); // --------------------------------------------------------------------------------------- // Write to file // -------------- for (int ii = 0; ii < nFiles; ii++) { std::string sfbuff; if(a == 0) sfbuff = "***"; // note if it's before or after an update (*** = before) else sfbuff = " "; // write running char cbuf[10]; sprintf(cbuf, "%i", PBs[i].running); sfbuff = sfbuff + cbuf; sfbuff = sfbuff + writeMessage(ii, i); // write _iSize strcpy(cbuf, ""); sprintf(cbuf, "%i", _iSize); sfbuff = sfbuff + " | _iSize: " + cbuf + "\n"; aprintf(ii, (char *)sfbuff.c_str()); } // -------------- } } // ============== //PanicAlert("Done now before: %i", numberOfPBs); // ======================================================================================= // Control how often the screen is updated, and then update the screen // -------------- if(a == 0) j++; if (m_frame->gUpdFreq > 0 && j > (200/m_frame->gUpdFreq)) { // ======================================================================================= // Move all items back. Vector1 is a vector1[NUMBER_OF_PBS][100] vector. // -------------- /* Move all items back like this: 1 to 2 2 3 3 ... */ for (int i = 0; i < NUMBER_OF_PBS; i++) { for (int j = 1; j < vectorLength; j++) { vector1.at(i).at(j-1) = vector1.at(i).at(j); } } // Save the latest value for (int i = 0; i < numberOfPBs; i++) { vector1.at(i).at(vectorLength-1) = PBs[i].running ? true : false; } // ============== // ======================================================================================= // Have a separate set for which ones to show // -------------- /* Move all items back like this: 1 to 2 2 3 3 ... */ for (int i = 0; i < NUMBER_OF_PBS; i++) { for (int j = 1; j < vectorLength2; j++) { vector2.at(i).at(j-1) = vector2.at(i).at(j); } } // Save the latest value for (int i = 0; i < numberOfPBs; i++) { vector2.at(i).at(vectorLength2-1) = PBs[i].running; } // ============== // ======================================================================================= // Count how many we have running now // -------------- int jj = 0; for (int i = 0; i < NUMBER_OF_PBS; i++) { for (int j = 0; j < vectorLength2-1; j++) { if (vector2.at(i).at(j) == 1) { jj++; } numberRunning.at(i) = jj; } } // ============== // ======================================================================================= // Write header // -------------- char buffer [1000] = ""; std::string sbuff; sbuff = writeTitle(m_frame->gPreset); // ============== // go through all running blocks for (int i = 0; i < numberOfPBs; i++) { // Prepare conditions. TODO: We use this in two places now, make it only one /**/if(Wii) Conditions = PrepareConditions(Wii, i, PBw); else Conditions = PrepareConditions(Wii, i, PBs); // Use the condition if (Conditions) { // Save playback history for the GUI debugger -------------------------- if(m_frame) { std::string guipr; // gui progress for (int j = 0; j < vectorLengthGUI; j++) { if(vector1.at(i).at(j) == 0) { guipr = guipr + "0"; } else { guipr = guipr + "1"; } } u32 run = atoi( guipr.c_str()); m_frame->m_GPRListView->m_CachedRegs[1][i] = run; guipr.clear(); } // Make the playback history (progress bar) to display in the console debugger for (int j = 0; j < vectorLength; j++) { if(vector1.at(i).at(j) == 0) { sbuff = sbuff + " "; } else { sprintf(buffer, "%c", 177); sbuff = sbuff + buffer; strcpy(buffer, ""); } } // --------- // Hopefully this is false if we don't have a debugging window and so it doesn't cause a crash if(m_frame) { m_frame->m_GPRListView->m_CachedRegs[2][i] = gsamplePos[i]; m_frame->m_GPRListView->m_CachedRegs[3][i] = gsampleEnd[i]; m_frame->m_GPRListView->m_CachedRegs[4][i] = gloopPos[i]; m_frame->m_GPRListView->m_CachedRegs[5][i] = gvolume_left[i]; m_frame->m_GPRListView->m_CachedRegs[6][i] = gvolume_right[i]; m_frame->m_GPRListView->m_CachedRegs[7][i] = glooping[i]; m_frame->m_GPRListView->m_CachedRegs[8][i] = gloop1[i]; m_frame->m_GPRListView->m_CachedRegs[9][i] = gloop2[i]; m_frame->m_GPRListView->m_CachedRegs[10][i] = gloop3[i]; m_frame->m_GPRListView->m_CachedRegs[11][i] = gis_stream[i]; m_frame->m_GPRListView->m_CachedRegs[12][i] = gaudioFormat[i]; m_frame->m_GPRListView->m_CachedRegs[13][i] = gsrc_type[i]; m_frame->m_GPRListView->m_CachedRegs[14][i] = gcoef[i]; m_frame->m_GPRListView->m_CachedRegs[15][i] = gfrac[i]; m_frame->m_GPRListView->m_CachedRegs[16][i] = gratio[i]; m_frame->m_GPRListView->m_CachedRegs[17][i] = gratiohi[i]; m_frame->m_GPRListView->m_CachedRegs[18][i] = gratiolo[i]; m_frame->m_GPRListView->m_CachedRegs[19][i] = gupdates1[i]; m_frame->m_GPRListView->m_CachedRegs[20][i] = gupdates2[i]; m_frame->m_GPRListView->m_CachedRegs[21][i] = gupdates3[i]; m_frame->m_GPRListView->m_CachedRegs[22][i] = gupdates4[i]; m_frame->m_GPRListView->m_CachedRegs[23][i] = gupdates5[i]; } // add new line sbuff = sbuff + writeMessage(m_frame->gPreset, i); strcpy(buffer, ""); sbuff = sbuff + "\n"; } // end of if (PBs[i].running) } // end of big loop - for (int i = 0; i < numberOfPBs; i++) // ======================================================================================= // Write global values // --------------- int nOfBlocks; if(Wii) nOfBlocks = (gLastBlock-m_addressPBs) / 256; else nOfBlocks = (gLastBlock-m_addressPBs) / 192; sprintf(buffer, "\nThe parameter blocks span from %08x to %08x | distance %i | num. of blocks %i | _iSize %i | numberOfPBs %i\n", m_addressPBs, gLastBlock, (gLastBlock-m_addressPBs), nOfBlocks, _iSize, numberOfPBs); sbuff = sbuff + buffer; strcpy(buffer, ""); // =============== // ======================================================================================= // Write settings // --------------- sprintf(buffer, "\nSettings: SSBM fix %i | SSBM rem1 %i | SSBM rem2 %i\nSequenced %i | Volume %i | Reset %i | Only looping %i | Save file %i\n", m_frame->gSSBM, m_frame->gSSBMremedy1, m_frame->gSSBMremedy2, m_frame->gSequenced, m_frame->gVolume, m_frame->gReset, m_frame->gOnlyLooping, m_frame->gSaveFile); sbuff = sbuff + buffer; strcpy(buffer, ""); // =============== // ======================================================================================= // Show update frequency // --------------- sbuff = sbuff + "\n"; if(!iupdonce) { viupd.at(0) = 1; viupd.at(1) = 1; viupd.at(2) = 1; iupdonce = true; } for (int i = 0; i < viupd.size(); i++) // 0, 1,..., 9 { if (i < viupd.size()-1) { viupd.at(viupd.size()-i-1) = viupd.at(viupd.size()-i-2); // move all forward } else { viupd.at(0) = viupd.at(viupd.size()-1); } // Correction if (viupd.at(viupd.size()-3) == 1 && viupd.at(viupd.size()-2) == 1 && viupd.at(viupd.size()-1) == 1) { viupd.at(0) = 0; } if(viupd.at(0) == 0 && viupd.at(1) == 1 && viupd.at(2) == 1 && viupd.at(3) == 0) { viupd.at(0) = 1; } } for (int i = 0; i < viupd.size(); i++) { if(viupd.at(i) == 0) sbuff = sbuff + " "; else sbuff = sbuff + "."; } // ================ // ======================================================================================= // Print // ---------------- ClearScreen(); wprintf("%s", sbuff.c_str()); sbuff.clear(); strcpy(buffer, ""); // ================ // New values are written so update - DISABLED - It flickered a lot, even worse than a // console window. So for now only the console windows is updated. /* if(m_frame) { m_frame->NotifyUpdate(); } */ k=0; j=0; } // end of if (j>20) } // end of function