2019-09-27 23:52:24 -07:00
using System ;
2021-07-18 09:44:23 -07:00
using System.Collections.Concurrent ;
2019-09-27 23:52:24 -07:00
using System.Collections.Generic ;
using System.IO ;
using System.Linq ;
2019-10-24 16:09:43 -04:00
using System.Text ;
2022-03-14 10:40:44 -07:00
using BurnOutSharp.ExecutableType.Microsoft.PE ;
2022-05-01 17:41:50 -07:00
using BurnOutSharp.Interfaces ;
2021-03-21 15:34:19 -07:00
using BurnOutSharp.Matching ;
2019-09-27 23:52:24 -07:00
namespace BurnOutSharp.ProtectionType
{
2022-10-11 22:00:30 -06:00
/// <summary>
/// CactusDataShield was a copy protection originally developed by Midbar Technologies, which was then purchased by Macrovision in 2002 (https://variety.com/2002/digital/news/macrovision-acquires-midbar-cuts-ttr-link-1117875824/).
2022-10-25 11:31:17 -06:00
/// Macrovision's product page for CDS: https://web.archive.org/web/20050215235405/http://www.macrovision.com/products/cds/index.shtml
2022-10-11 22:00:30 -06:00
/// CDS-100 appears to function by attempting to prevent dumping/ripping the discs protected with it.
/// CDS-200+ uses a dedicated audio player to play the music "legitimately".
/// Patent relating to CDS-100: https://patents.google.com/patent/US6425098B1/
/// Known CDS versions:
/// CDS-100 ("The Loveparade Compilation 2001" by various artists (Barcode 74321 86986 2) (Likely Discogs Release Code [r155963]) and "World Of Our Own" (Limited Edition) by Westlife (Barcode 7 43218 98572 0) (Discogs Release Code [r1357706])).
/// CDS-200 (PlayJ) (("Volumia!" by Puur (Barcode 7 43218 63282 2) (Discogs Release Code [r795427])) (Confirmed to be CDS-200 from https://www.cdrinfo.com/d7/content/cactus-data-shield-200?page=2).
/// CDS200.0.4 - 3.0 build 16a (Redump entry 95036)
/// CDS200.0.4 - 3.0 build 16c ("TMF Hitzone 20" by various artists (Barcode 7 31458 37062 8)).
/// CDS200.0.4 - 4.1 build 2a ("Ich Habe Einen Traum" by Uwe Busse (Barcode 9 002723 251203)).
/// CDS200.0.4 - 4.1 build 2e ("Hallucinations" by David Usher (Barcode 7 24359 30322 2)).
2022-10-27 17:40:16 -06:00
/// CDS200.5.11.90 - 5.10.090 ("Finn 5 Fel!" by Gyllene Tider (Barcode 7 24357 10922 2)).
2022-10-11 22:00:30 -06:00
/// CDS-300
/// Further information:
/// https://www.cdrinfo.com/d7/content/cactus-data-shield-200
/// https://www.cdmediaworld.com/hardware/cdrom/cd_protections_cactus_data_shield.shtml
/// </summary>
2022-05-01 17:23:00 -07:00
public class CactusDataShield : IContentCheck , IPathCheck , IPortableExecutableCheck
2019-09-27 23:52:24 -07:00
{
2021-08-25 19:37:32 -07:00
/// <inheritdoc/>
2022-05-01 17:17:15 -07:00
public string CheckContents ( string file , byte [ ] fileContent , bool includeDebug )
2022-03-14 11:20:11 -07:00
{
// TODO: Limit these checks to Mac binaries
// TODO: Obtain a sample to find where this string is in a typical executable
if ( includeDebug )
{
2022-03-14 11:52:49 -07:00
var contentMatchSets = new List < ContentMatchSet >
{
// CDSPlayer
new ContentMatchSet ( new byte? [ ] { 0x43 , 0x44 , 0x53 , 0x50 , 0x6C , 0x61 , 0x79 , 0x65 , 0x72 } , "Cactus Data Shield 200" ) ,
// yucca.cds
new ContentMatchSet ( new byte? [ ] { 0x79 , 0x75 , 0x63 , 0x63 , 0x61 , 0x2E , 0x63 , 0x64 , 0x73 } , "Cactus Data Shield 200" ) ,
} ;
2022-03-14 11:20:11 -07:00
if ( contentMatchSets ! = null & & contentMatchSets . Any ( ) )
return MatchUtil . GetFirstMatch ( file , fileContent , contentMatchSets , includeDebug ) ;
}
return null ;
}
/// <inheritdoc/>
2022-05-01 17:17:15 -07:00
public string CheckPortableExecutable ( string file , PortableExecutable pex , bool includeDebug )
2021-08-29 21:07:26 -07:00
{
// Get the sections from the executable, if possible
var sections = pex ? . SectionTable ;
if ( sections = = null )
return null ;
// Get the .data section, if it exists
2021-09-11 16:47:25 -07:00
if ( pex . DataSectionRaw ! = null )
2021-08-29 21:07:26 -07:00
{
var matchers = new List < ContentMatchSet >
{
// \*.CDS
2021-09-11 16:47:25 -07:00
new ContentMatchSet ( new byte? [ ] { 0x5C , 0x2A , 0x2E , 0x43 , 0x44 , 0x53 } , "Cactus Data Shield 200" ) ,
2021-08-29 21:07:26 -07:00
// DATA.CDS
2021-09-11 16:47:25 -07:00
new ContentMatchSet ( new byte? [ ] { 0x44 , 0x41 , 0x54 , 0x41 , 0x2E , 0x43 , 0x44 , 0x53 } , "Cactus Data Shield 200" ) ,
2021-08-29 21:07:26 -07:00
} ;
2021-09-11 16:47:25 -07:00
string match = MatchUtil . GetFirstMatch ( file , pex . DataSectionRaw , matchers , includeDebug ) ;
2021-08-29 21:07:26 -07:00
if ( ! string . IsNullOrWhiteSpace ( match ) )
return match ;
}
2022-10-11 22:00:30 -06:00
// Get the .rsrc section, if it exists
var rsrcSectionRaw = pex . ReadRawSection ( ".rsrc" , first : false ) ;
if ( rsrcSectionRaw ! = null )
{
var matchers = new List < ContentMatchSet >
{
// CactusPJ
// Found in "Volumia!" by Puur (Barcode 7 43218 63282 2) (Discogs Release Code [r795427]).
// Modified version of the PlayJ Music Player specificaly for CDS, as indicated by the About page present when running the executable.
2022-10-13 23:14:50 -06:00
new ContentMatchSet ( new byte? [ ] { 0x43 , 0x61 , 0x63 , 0x74 , 0x75 , 0x73 , 0x50 , 0x4A } , "PlayJ Music Player (Cactus Data Shield 200)" ) ,
2022-10-11 22:00:30 -06:00
} ;
string match = MatchUtil . GetFirstMatch ( file , rsrcSectionRaw , matchers , includeDebug ) ;
if ( ! string . IsNullOrWhiteSpace ( match ) )
return match ;
}
2021-08-29 21:07:26 -07:00
return null ;
}
2019-09-27 23:52:24 -07:00
2021-02-26 00:32:09 -08:00
/// <inheritdoc/>
2021-07-18 09:44:23 -07:00
public ConcurrentQueue < string > CheckDirectoryPath ( string path , IEnumerable < string > files )
2019-09-27 23:52:24 -07:00
{
2021-03-22 23:02:01 -07:00
// TODO: Verify if these are OR or AND
var matchers = new List < PathMatchSet >
2019-09-27 23:52:24 -07:00
{
2022-10-11 22:00:30 -06:00
// Found in "Volumia!" by Puur (Barcode 7 43218 63282 2) (Discogs Release Code [r795427]).
// Modified version of the PlayJ Music Player specificaly for CDS, as indicated by the About page present when running the executable.
// The file "DATA16.BML" is also present on this disc but the name is too generic to check for.
2022-10-13 23:14:50 -06:00
new PathMatchSet ( new PathMatch ( "CACTUSPJ.exe" , useEndsWith : true ) , "PlayJ Music Player (Cactus Data Shield 200)" ) ,
2022-10-11 22:00:30 -06:00
// Found in "Volumia!" by Puur (Barcode 7 43218 63282 2) (Discogs Release Code [r795427]).
// In "Volumina! - Puur" (7 43218 63282 2), this file is composed of multiple PLJ files combined together.
// In later versions, this file is a padded dummy file. ("Ich Habe Einen Traum" by Uwe Busse (Barcode 9 002723 251203)).
2022-10-13 23:14:50 -06:00
new PathMatchSet ( new PathMatch ( "YUCCA.CDS" , useEndsWith : true ) , "Cactus Data Shield 200" ) ,
2022-10-11 22:00:30 -06:00
// TODO: Find samples of the following:
2022-10-13 23:14:50 -06:00
new PathMatchSet ( new PathMatch ( "CDSPlayer.app" , useEndsWith : true ) , GetVersion , "Cactus Data Shield" ) ,
new PathMatchSet ( new PathMatch ( "wmmp.exe" , useEndsWith : true ) , GetVersion , "Cactus Data Shield" ) ,
2022-08-21 21:20:28 -06:00
// Present on CDS-300, as well as SafeDisc. This is likely due to both protections being created by Macrovision.
new PathMatchSet ( new PathMatch ( "00000001.TMP" , useEndsWith : true ) , Get00000001TMPVersion , "Cactus Data Shield 300 (Confirm presence of other CDS-300 files)" ) ,
2021-03-22 23:02:01 -07:00
} ;
2021-03-19 15:41:49 -07:00
2021-03-23 13:35:12 -07:00
return MatchUtil . GetAllMatches ( files , matchers , any : true ) ;
2021-03-19 15:41:49 -07:00
}
/// <inheritdoc/>
public string CheckFilePath ( string path )
{
2021-03-22 23:02:01 -07:00
var matchers = new List < PathMatchSet >
{
2022-10-11 22:00:30 -06:00
// Found in "Volumia!" by Puur (Barcode 7 43218 63282 2) (Discogs Release Code [r795427]).
// Modified version of the PlayJ Music Player specificaly for CDS, as indicated by the About page present when running the executable.
// The file "DATA16.BML" is also present on this disc but the name is too generic to check for.
2022-10-13 23:14:50 -06:00
new PathMatchSet ( new PathMatch ( "CACTUSPJ.exe" , useEndsWith : true ) , "PlayJ Music Player (Cactus Data Shield 200)" ) ,
2022-10-11 22:00:30 -06:00
// Found in "Volumia!" by Puur (Barcode 7 43218 63282 2) (Discogs Release Code [r795427]),
// In "Volumia! - Puur", this file is composed of multiple PLJ files combined together.
// In later versions, this file is a padded dummy file. ("Ich Habe Einen Traum" by Uwe Busse (Barcode 9 002723 251203)).
2022-10-13 23:14:50 -06:00
new PathMatchSet ( new PathMatch ( "YUCCA.CDS" , useEndsWith : true ) , "Cactus Data Shield 200" ) ,
2022-10-11 22:00:30 -06:00
// TODO: Find samples of the following:
2021-03-22 23:02:01 -07:00
new PathMatchSet ( new PathMatch ( "CDSPlayer.app" , useEndsWith : true ) , "Cactus Data Shield 200" ) ,
new PathMatchSet ( new PathMatch ( "wmmp.exe" , useEndsWith : true ) , "Cactus Data Shield 200" ) ,
2022-08-21 21:20:28 -06:00
// Present on CDS-300, as well as SafeDisc. This is likely due to both protections being created by Macrovision.
2022-08-21 20:36:37 -07:00
new PathMatchSet ( new PathMatch ( "00000001.TMP" , useEndsWith : true ) , Get00000001TMPVersion , "Cactus Data Shield 300" ) ,
2021-03-22 23:02:01 -07:00
} ;
return MatchUtil . GetFirstMatch ( path , matchers , any : true ) ;
}
2022-08-21 21:20:28 -06:00
public static string Get00000001TMPVersion ( string firstMatchedString , IEnumerable < string > files )
{
if ( string . IsNullOrEmpty ( firstMatchedString ) | | ! File . Exists ( firstMatchedString ) )
return string . Empty ;
// This file is present on both CDS-300 and SafeDisc.
// Only one specific file size appears to be associated with CDS-300, so any files with a differing file size are discarded. If it is the correct file size, return it as valid.
FileInfo fi = new FileInfo ( firstMatchedString ) ;
switch ( fi . Length )
{
case 2_048 :
2022-08-21 20:36:37 -07:00
return "(Confirm presence of other CDS-300 files)" ;
2022-08-21 21:20:28 -06:00
default :
return null ;
}
}
2022-10-27 17:40:16 -06:00
// TODO: Simplify version checking.
2021-03-22 23:02:01 -07:00
public static string GetVersion ( string firstMatchedString , IEnumerable < string > files )
{
// Find the version.txt file first
string versionPath = files . FirstOrDefault ( f = > Path . GetFileName ( f ) . Equals ( "version.txt" , StringComparison . OrdinalIgnoreCase ) ) ;
if ( ! string . IsNullOrWhiteSpace ( versionPath ) )
2019-09-27 23:52:24 -07:00
{
2021-03-22 23:02:01 -07:00
string version = GetInternalVersion ( versionPath ) ;
if ( ! string . IsNullOrWhiteSpace ( version ) )
return version ;
2019-09-27 23:52:24 -07:00
}
2021-03-22 23:02:01 -07:00
return "200" ;
2019-09-27 23:52:24 -07:00
}
2020-10-27 14:37:14 -07:00
2021-03-22 23:02:01 -07:00
private static string GetInternalVersion ( string path )
2020-10-27 14:37:14 -07:00
{
if ( ! File . Exists ( path ) )
return null ;
try
{
using ( var sr = new StreamReader ( path , Encoding . Default ) )
{
2020-10-27 17:01:33 -07:00
return $"{sr.ReadLine().Substring(3)} ({sr.ReadLine()})" ;
2020-10-27 14:37:14 -07:00
}
}
catch
{
return null ;
}
}
2019-09-27 23:52:24 -07:00
}
}