CCdInfo* CCdIoSupport::GetCdInfo(char* cDeviceFileName) { CSingleLock lock(*m_cdio); char* source_name; if(cDeviceFileName == NULL) source_name = m_cdio->GetDeviceFileName(); else source_name = cDeviceFileName; cdio = ::cdio_open(source_name, DRIVER_UNKNOWN); if (cdio == NULL) { CLog::Log(LOGERROR, "%s: Error in automatically selecting driver with input", __FUNCTION__); return NULL; } bool bIsCDRom = true; m_nFirstTrackNum = ::cdio_get_first_track_num(cdio); if (m_nFirstTrackNum == CDIO_INVALID_TRACK) { #if !defined(TARGET_DARWIN) ::cdio_destroy(cdio); return NULL; #else m_nFirstTrackNum = 1; bIsCDRom = false; #endif } m_nNumTracks = ::cdio_get_num_tracks(cdio); if (m_nNumTracks == CDIO_INVALID_TRACK) { #if !defined(TARGET_DARWIN) ::cdio_destroy(cdio); return NULL; #else m_nNumTracks = 1; bIsCDRom = false; #endif } CCdInfo* info = new CCdInfo; info->SetFirstTrack( m_nFirstTrackNum ); info->SetTrackCount( m_nNumTracks ); for (i = m_nFirstTrackNum; i <= CDIO_CDROM_LEADOUT_TRACK; i++) { msf_t msf; if (bIsCDRom && !::cdio_get_track_msf(cdio, i, &msf)) { trackinfo ti; ti.nfsInfo = FS_UNKNOWN; ti.ms_offset = 0; ti.isofs_size = 0; ti.nJolietLevel = 0; ti.nFrames = 0; info->SetTrackInformation( i, ti ); CLog::Log(LOGDEBUG, "cdio_track_msf for track %i failed, I give up.", i); delete info; ::cdio_destroy(cdio); return NULL; } trackinfo ti; if (bIsCDRom && TRACK_FORMAT_AUDIO == ::cdio_get_track_format(cdio, i)) { m_nNumAudio++; ti.nfsInfo = FS_NO_DATA; m_nFs = FS_NO_DATA; int temp1 = ::cdio_get_track_lba(cdio, i) - CDIO_PREGAP_SECTORS; int temp2 = ::cdio_get_track_lba(cdio, i + 1) - CDIO_PREGAP_SECTORS; // the length is the address of the second track minus the address of the first track temp2 -= temp1; // temp2 now has length of track1 in frames ti.nMins = temp2 / (60 * 75); // calculate the number of minutes temp2 %= 60 * 75; // calculate the left-over frames ti.nSecs = temp2 / 75; // calculate the number of seconds if ( -1 == m_nFirstAudio) m_nFirstAudio = i; // Make sure that we have the Disc related info available if (i == 1) { xbmc_cdtext_t xcdt; GetCdTextInfo(xcdt, 0); info->SetDiscCDTextInformation( xcdt ); } // Get this tracks info GetCdTextInfo(ti.cdtext, i); } else { m_nNumData++; if ( -1 == m_nFirstData) m_nFirstData = i; } ti.nfsInfo = FS_NO_DATA; ti.ms_offset = 0; ti.isofs_size = 0; ti.nJolietLevel = 0; ti.nFrames = ::cdio_get_track_lba(cdio, i); ti.nMins = 0; ti.nSecs = 0; info->SetTrackInformation( i, ti ); /* skip to leadout? */ if (i == m_nNumTracks) i = CDIO_CDROM_LEADOUT_TRACK; } info->SetCddbDiscId( CddbDiscId() ); info->SetDiscLength( ::cdio_get_track_lba(cdio, CDIO_CDROM_LEADOUT_TRACK) / CDIO_CD_FRAMES_PER_SEC ); info->SetAudioTrackCount( m_nNumAudio ); info->SetDataTrackCount( m_nNumData ); info->SetFirstAudioTrack( m_nFirstAudio ); info->SetFirstDataTrack( m_nFirstData ); CLog::Log(LOGINFO, "CD Analysis Report"); CLog::Log(LOGINFO, STRONG); /* try to find out what sort of CD we have */ if (0 == m_nNumData) { /* no data track, may be a "real" audio CD or hidden track CD */ msf_t msf; ::cdio_get_track_msf(cdio, 1, &msf); m_nStartTrack = ::cdio_msf_to_lsn(&msf); /* CD-I/Ready says start_track <= 30*75 then CDDA */ if (m_nStartTrack > 100 /* 100 is just a guess */) { m_nFs = GuessFilesystem(0, 1); if ((m_nFs & FS_MASK) != FS_UNKNOWN) m_nFs |= HIDDEN_TRACK; else { m_nFs &= ~FS_MASK; /* del filesystem info */ CLog::Log(LOGDEBUG, "Oops: %i unused sectors at start, but hidden track check failed.", m_nStartTrack); } } PrintAnalysis(m_nFs, m_nNumAudio); } else { /* we have data track(s) */ for (j = 2, i = m_nFirstData; i <= m_nNumTracks; i++) { msf_t msf; track_format_t track_format = ::cdio_get_track_format(cdio, i); ::cdio_get_track_msf(cdio, i, &msf); switch ( track_format ) { case TRACK_FORMAT_AUDIO: { trackinfo ti; ti.nfsInfo = FS_NO_DATA; m_nFs = FS_NO_DATA; ti.ms_offset = 0; ti.isofs_size = 0; ti.nJolietLevel = 0; ti.nFrames = ::cdio_get_track_lba(cdio, i); ti.nMins = 0; ti.nSecs = 0; info->SetTrackInformation( i + 1, ti ); } case TRACK_FORMAT_ERROR: break; case TRACK_FORMAT_CDI: case TRACK_FORMAT_XA: case TRACK_FORMAT_DATA: case TRACK_FORMAT_PSX: break; } m_nStartTrack = (i == 1) ? 0 : ::cdio_msf_to_lsn(&msf); /* save the start of the data area */ if (i == m_nFirstData) m_nDataStart = m_nStartTrack; /* skip tracks which belong to the current walked session */ if (m_nStartTrack < m_nDataStart + m_nIsofsSize) continue; m_nFs = GuessFilesystem(m_nStartTrack, i); trackinfo ti; ti.nfsInfo = m_nFs; ti.ms_offset = m_nMsOffset; ti.isofs_size = m_nIsofsSize; ti.nJolietLevel = m_nJolietLevel; ti.nFrames = ::cdio_get_track_lba(cdio, i); ti.nMins = 0; ti.nSecs = 0; info->SetDiscLabel(m_strDiscLabel); if (i > 1) { /* track is beyond last session -> new session found */ m_nMsOffset = m_nStartTrack; CLog::Log(LOGINFO, "Session #%d starts at track %2i, LSN: %6i," " ISO 9660 blocks: %6i", j++, i, m_nStartTrack, m_nIsofsSize); CLog::Log(LOGINFO, "ISO 9660: %i blocks, label '%.32s'\n", m_nIsofsSize, buffer[0] + 40); m_nFs |= MULTISESSION; ti.nfsInfo = m_nFs; } else { PrintAnalysis(m_nFs, m_nNumAudio); } info->SetTrackInformation( i, ti ); } } ::cdio_destroy( cdio ); return info; }
void CGUIWindowMusicSongs::GetContextButtons(int itemNumber, CContextButtons &buttons) { CFileItemPtr item; if (itemNumber >= 0 && itemNumber < m_vecItems->Size()) item = m_vecItems->Get(itemNumber); if (item) { // are we in the playlists location? bool inPlaylists = m_vecItems->GetPath().Equals(CUtil::MusicPlaylistsLocation()) || m_vecItems->GetPath().Equals("special://musicplaylists/"); if (m_vecItems->IsVirtualDirectoryRoot()) { // get the usual music shares, and anything for all media windows CGUIDialogContextMenu::GetContextButtons("music", item, buttons); #ifdef HAS_DVD_DRIVE // enable Rip CD an audio disc if (g_mediaManager.IsDiscInDrive() && item->IsCDDA()) { // those cds can also include Audio Tracks: CDExtra and MixedMode! CCdInfo *pCdInfo = g_mediaManager.GetCdInfo(); if (pCdInfo->IsAudio(1) || pCdInfo->IsCDExtra(1) || pCdInfo->IsMixedMode(1)) buttons.Add(CONTEXT_BUTTON_RIP_CD, 600); } #endif CGUIMediaWindow::GetContextButtons(itemNumber, buttons); } else { CGUIWindowMusicBase::GetContextButtons(itemNumber, buttons); if (item->GetProperty("pluginreplacecontextitems").asBoolean()) return; if (!item->IsPlayList()) { if (item->IsAudio() && !item->IsLastFM()) buttons.Add(CONTEXT_BUTTON_SONG_INFO, 658); // Song Info else if (!item->IsParentFolder() && !item->IsLastFM() && !item->GetPath().Left(3).Equals("new") && item->m_bIsFolder) { #if 0 if (m_musicdatabase.GetAlbumIdByPath(item->GetPath()) > -1) #endif buttons.Add(CONTEXT_BUTTON_INFO, 13351); // Album Info } } #ifdef HAS_DVD_DRIVE // enable Rip CD Audio or Track button if we have an audio disc if (g_mediaManager.IsDiscInDrive() && m_vecItems->IsCDDA()) { // those cds can also include Audio Tracks: CDExtra and MixedMode! CCdInfo *pCdInfo = g_mediaManager.GetCdInfo(); if (pCdInfo->IsAudio(1) || pCdInfo->IsCDExtra(1) || pCdInfo->IsMixedMode(1)) buttons.Add(CONTEXT_BUTTON_RIP_TRACK, 610); } #endif // enable CDDB lookup if the current dir is CDDA if (g_mediaManager.IsDiscInDrive() && m_vecItems->IsCDDA() && (g_settings.GetCurrentProfile().canWriteDatabases() || g_passwordManager.bMasterUser)) { buttons.Add(CONTEXT_BUTTON_CDDB, 16002); } if (!item->IsParentFolder() && !item->IsReadOnly()) { // either we're at the playlist location or its been explicitly allowed if (inPlaylists || g_guiSettings.GetBool("filelists.allowfiledeletion")) { buttons.Add(CONTEXT_BUTTON_DELETE, 117); buttons.Add(CONTEXT_BUTTON_RENAME, 118); } } } // Add the scan button(s) if (g_application.IsMusicScanning()) buttons.Add(CONTEXT_BUTTON_STOP_SCANNING, 13353); // Stop Scanning else if (!inPlaylists && !m_vecItems->IsInternetStream() && !item->IsLastFM() && !item->GetPath().Equals("add") && !item->IsParentFolder() && !item->IsPlugin() && (g_settings.GetCurrentProfile().canWriteDatabases() || g_passwordManager.bMasterUser)) { buttons.Add(CONTEXT_BUTTON_SCAN, 13352); } if (item->IsPlugin() || item->IsScript() || m_vecItems->IsPlugin()) buttons.Add(CONTEXT_BUTTON_PLUGIN_SETTINGS, 1045); } if (!m_vecItems->IsVirtualDirectoryRoot()) buttons.Add(CONTEXT_BUTTON_SWITCH_MEDIA, 523); CGUIWindowMusicBase::GetNonContextButtons(buttons); }