TSPlaylistFile.cs 42 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282
  1. //============================================================================
  2. // BDInfo - Blu-ray Video and Audio Analysis Tool
  3. // Copyright © 2010 Cinema Squid
  4. //
  5. // This library is free software; you can redistribute it and/or
  6. // modify it under the terms of the GNU Lesser General Public
  7. // License as published by the Free Software Foundation; either
  8. // version 2.1 of the License, or (at your option) any later version.
  9. //
  10. // This library is distributed in the hope that it will be useful,
  11. // but WITHOUT ANY WARRANTY; without even the implied warranty of
  12. // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
  13. // Lesser General Public License for more details.
  14. //
  15. // You should have received a copy of the GNU Lesser General Public
  16. // License along with this library; if not, write to the Free Software
  17. // Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
  18. //=============================================================================
  19. #undef DEBUG
  20. using System;
  21. using System.Collections.Generic;
  22. using System.IO;
  23. using System.Text;
  24. using MediaBrowser.Model.IO;
  25. namespace BDInfo
  26. {
  27. public class TSPlaylistFile
  28. {
  29. private FileSystemMetadata FileInfo = null;
  30. public string FileType = null;
  31. public bool IsInitialized = false;
  32. public string Name = null;
  33. public BDROM BDROM = null;
  34. public bool HasHiddenTracks = false;
  35. public bool HasLoops = false;
  36. public bool IsCustom = false;
  37. public List<double> Chapters = new List<double>();
  38. public Dictionary<ushort, TSStream> Streams =
  39. new Dictionary<ushort, TSStream>();
  40. public Dictionary<ushort, TSStream> PlaylistStreams =
  41. new Dictionary<ushort, TSStream>();
  42. public List<TSStreamClip> StreamClips =
  43. new List<TSStreamClip>();
  44. public List<Dictionary<ushort, TSStream>> AngleStreams =
  45. new List<Dictionary<ushort, TSStream>>();
  46. public List<Dictionary<double, TSStreamClip>> AngleClips =
  47. new List<Dictionary<double, TSStreamClip>>();
  48. public int AngleCount = 0;
  49. public List<TSStream> SortedStreams =
  50. new List<TSStream>();
  51. public List<TSVideoStream> VideoStreams =
  52. new List<TSVideoStream>();
  53. public List<TSAudioStream> AudioStreams =
  54. new List<TSAudioStream>();
  55. public List<TSTextStream> TextStreams =
  56. new List<TSTextStream>();
  57. public List<TSGraphicsStream> GraphicsStreams =
  58. new List<TSGraphicsStream>();
  59. public TSPlaylistFile(BDROM bdrom,
  60. FileSystemMetadata fileInfo)
  61. {
  62. BDROM = bdrom;
  63. FileInfo = fileInfo;
  64. Name = fileInfo.Name.ToUpper();
  65. }
  66. public TSPlaylistFile(BDROM bdrom,
  67. string name,
  68. List<TSStreamClip> clips)
  69. {
  70. BDROM = bdrom;
  71. Name = name;
  72. IsCustom = true;
  73. foreach (var clip in clips)
  74. {
  75. var newClip = new TSStreamClip(
  76. clip.StreamFile, clip.StreamClipFile);
  77. newClip.Name = clip.Name;
  78. newClip.TimeIn = clip.TimeIn;
  79. newClip.TimeOut = clip.TimeOut;
  80. newClip.Length = newClip.TimeOut - newClip.TimeIn;
  81. newClip.RelativeTimeIn = TotalLength;
  82. newClip.RelativeTimeOut = newClip.RelativeTimeIn + newClip.Length;
  83. newClip.AngleIndex = clip.AngleIndex;
  84. newClip.Chapters.Add(clip.TimeIn);
  85. StreamClips.Add(newClip);
  86. if (newClip.AngleIndex > AngleCount)
  87. {
  88. AngleCount = newClip.AngleIndex;
  89. }
  90. if (newClip.AngleIndex == 0)
  91. {
  92. Chapters.Add(newClip.RelativeTimeIn);
  93. }
  94. }
  95. LoadStreamClips();
  96. IsInitialized = true;
  97. }
  98. public override string ToString()
  99. {
  100. return Name;
  101. }
  102. public ulong InterleavedFileSize
  103. {
  104. get
  105. {
  106. ulong size = 0;
  107. foreach (var clip in StreamClips)
  108. {
  109. size += clip.InterleavedFileSize;
  110. }
  111. return size;
  112. }
  113. }
  114. public ulong FileSize
  115. {
  116. get
  117. {
  118. ulong size = 0;
  119. foreach (var clip in StreamClips)
  120. {
  121. size += clip.FileSize;
  122. }
  123. return size;
  124. }
  125. }
  126. public double TotalLength
  127. {
  128. get
  129. {
  130. double length = 0;
  131. foreach (var clip in StreamClips)
  132. {
  133. if (clip.AngleIndex == 0)
  134. {
  135. length += clip.Length;
  136. }
  137. }
  138. return length;
  139. }
  140. }
  141. public double TotalAngleLength
  142. {
  143. get
  144. {
  145. double length = 0;
  146. foreach (var clip in StreamClips)
  147. {
  148. length += clip.Length;
  149. }
  150. return length;
  151. }
  152. }
  153. public ulong TotalSize
  154. {
  155. get
  156. {
  157. ulong size = 0;
  158. foreach (var clip in StreamClips)
  159. {
  160. if (clip.AngleIndex == 0)
  161. {
  162. size += clip.PacketSize;
  163. }
  164. }
  165. return size;
  166. }
  167. }
  168. public ulong TotalAngleSize
  169. {
  170. get
  171. {
  172. ulong size = 0;
  173. foreach (var clip in StreamClips)
  174. {
  175. size += clip.PacketSize;
  176. }
  177. return size;
  178. }
  179. }
  180. public ulong TotalBitRate
  181. {
  182. get
  183. {
  184. if (TotalLength > 0)
  185. {
  186. return (ulong)Math.Round(((TotalSize * 8.0) / TotalLength));
  187. }
  188. return 0;
  189. }
  190. }
  191. public ulong TotalAngleBitRate
  192. {
  193. get
  194. {
  195. if (TotalAngleLength > 0)
  196. {
  197. return (ulong)Math.Round(((TotalAngleSize * 8.0) / TotalAngleLength));
  198. }
  199. return 0;
  200. }
  201. }
  202. public void Scan(
  203. Dictionary<string, TSStreamFile> streamFiles,
  204. Dictionary<string, TSStreamClipFile> streamClipFiles)
  205. {
  206. Stream fileStream = null;
  207. BinaryReader fileReader = null;
  208. try
  209. {
  210. Streams.Clear();
  211. StreamClips.Clear();
  212. fileStream = File.OpenRead(FileInfo.FullName);
  213. fileReader = new BinaryReader(fileStream);
  214. byte[] data = new byte[fileStream.Length];
  215. int dataLength = fileReader.Read(data, 0, data.Length);
  216. int pos = 0;
  217. FileType = ReadString(data, 8, ref pos);
  218. if (FileType != "MPLS0100" && FileType != "MPLS0200")
  219. {
  220. throw new Exception(string.Format(
  221. "Playlist {0} has an unknown file type {1}.",
  222. FileInfo.Name, FileType));
  223. }
  224. int playlistOffset = ReadInt32(data, ref pos);
  225. int chaptersOffset = ReadInt32(data, ref pos);
  226. int extensionsOffset = ReadInt32(data, ref pos);
  227. pos = playlistOffset;
  228. int playlistLength = ReadInt32(data, ref pos);
  229. int playlistReserved = ReadInt16(data, ref pos);
  230. int itemCount = ReadInt16(data, ref pos);
  231. int subitemCount = ReadInt16(data, ref pos);
  232. var chapterClips = new List<TSStreamClip>();
  233. for (int itemIndex = 0; itemIndex < itemCount; itemIndex++)
  234. {
  235. int itemStart = pos;
  236. int itemLength = ReadInt16(data, ref pos);
  237. string itemName = ReadString(data, 5, ref pos);
  238. string itemType = ReadString(data, 4, ref pos);
  239. TSStreamFile streamFile = null;
  240. string streamFileName = string.Format(
  241. "{0}.M2TS", itemName);
  242. if (streamFiles.ContainsKey(streamFileName))
  243. {
  244. streamFile = streamFiles[streamFileName];
  245. }
  246. if (streamFile == null)
  247. {
  248. // Error condition
  249. }
  250. TSStreamClipFile streamClipFile = null;
  251. string streamClipFileName = string.Format(
  252. "{0}.CLPI", itemName);
  253. if (streamClipFiles.ContainsKey(streamClipFileName))
  254. {
  255. streamClipFile = streamClipFiles[streamClipFileName];
  256. }
  257. if (streamClipFile == null)
  258. {
  259. throw new Exception(string.Format(
  260. "Playlist {0} referenced missing file {1}.",
  261. FileInfo.Name, streamFileName));
  262. }
  263. pos += 1;
  264. int multiangle = (data[pos] >> 4) & 0x01;
  265. int condition = data[pos] & 0x0F;
  266. pos += 2;
  267. int inTime = ReadInt32(data, ref pos);
  268. if (inTime < 0) inTime &= 0x7FFFFFFF;
  269. double timeIn = (double)inTime / 45000;
  270. int outTime = ReadInt32(data, ref pos);
  271. if (outTime < 0) outTime &= 0x7FFFFFFF;
  272. double timeOut = (double)outTime / 45000;
  273. var streamClip = new TSStreamClip(
  274. streamFile, streamClipFile);
  275. streamClip.Name = streamFileName; //TODO
  276. streamClip.TimeIn = timeIn;
  277. streamClip.TimeOut = timeOut;
  278. streamClip.Length = streamClip.TimeOut - streamClip.TimeIn;
  279. streamClip.RelativeTimeIn = TotalLength;
  280. streamClip.RelativeTimeOut = streamClip.RelativeTimeIn + streamClip.Length;
  281. StreamClips.Add(streamClip);
  282. chapterClips.Add(streamClip);
  283. pos += 12;
  284. if (multiangle > 0)
  285. {
  286. int angles = data[pos];
  287. pos += 2;
  288. for (int angle = 0; angle < angles - 1; angle++)
  289. {
  290. string angleName = ReadString(data, 5, ref pos);
  291. string angleType = ReadString(data, 4, ref pos);
  292. pos += 1;
  293. TSStreamFile angleFile = null;
  294. string angleFileName = string.Format(
  295. "{0}.M2TS", angleName);
  296. if (streamFiles.ContainsKey(angleFileName))
  297. {
  298. angleFile = streamFiles[angleFileName];
  299. }
  300. if (angleFile == null)
  301. {
  302. throw new Exception(string.Format(
  303. "Playlist {0} referenced missing angle file {1}.",
  304. FileInfo.Name, angleFileName));
  305. }
  306. TSStreamClipFile angleClipFile = null;
  307. string angleClipFileName = string.Format(
  308. "{0}.CLPI", angleName);
  309. if (streamClipFiles.ContainsKey(angleClipFileName))
  310. {
  311. angleClipFile = streamClipFiles[angleClipFileName];
  312. }
  313. if (angleClipFile == null)
  314. {
  315. throw new Exception(string.Format(
  316. "Playlist {0} referenced missing angle file {1}.",
  317. FileInfo.Name, angleClipFileName));
  318. }
  319. var angleClip =
  320. new TSStreamClip(angleFile, angleClipFile);
  321. angleClip.AngleIndex = angle + 1;
  322. angleClip.TimeIn = streamClip.TimeIn;
  323. angleClip.TimeOut = streamClip.TimeOut;
  324. angleClip.RelativeTimeIn = streamClip.RelativeTimeIn;
  325. angleClip.RelativeTimeOut = streamClip.RelativeTimeOut;
  326. angleClip.Length = streamClip.Length;
  327. StreamClips.Add(angleClip);
  328. }
  329. if (angles - 1 > AngleCount) AngleCount = angles - 1;
  330. }
  331. int streamInfoLength = ReadInt16(data, ref pos);
  332. pos += 2;
  333. int streamCountVideo = data[pos++];
  334. int streamCountAudio = data[pos++];
  335. int streamCountPG = data[pos++];
  336. int streamCountIG = data[pos++];
  337. int streamCountSecondaryAudio = data[pos++];
  338. int streamCountSecondaryVideo = data[pos++];
  339. int streamCountPIP = data[pos++];
  340. pos += 5;
  341. #if DEBUG
  342. Debug.WriteLine(string.Format(
  343. "{0} : {1} -> V:{2} A:{3} PG:{4} IG:{5} 2A:{6} 2V:{7} PIP:{8}",
  344. Name, streamFileName, streamCountVideo, streamCountAudio, streamCountPG, streamCountIG,
  345. streamCountSecondaryAudio, streamCountSecondaryVideo, streamCountPIP));
  346. #endif
  347. for (int i = 0; i < streamCountVideo; i++)
  348. {
  349. var stream = CreatePlaylistStream(data, ref pos);
  350. if (stream != null) PlaylistStreams[stream.PID] = stream;
  351. }
  352. for (int i = 0; i < streamCountAudio; i++)
  353. {
  354. var stream = CreatePlaylistStream(data, ref pos);
  355. if (stream != null) PlaylistStreams[stream.PID] = stream;
  356. }
  357. for (int i = 0; i < streamCountPG; i++)
  358. {
  359. var stream = CreatePlaylistStream(data, ref pos);
  360. if (stream != null) PlaylistStreams[stream.PID] = stream;
  361. }
  362. for (int i = 0; i < streamCountIG; i++)
  363. {
  364. var stream = CreatePlaylistStream(data, ref pos);
  365. if (stream != null) PlaylistStreams[stream.PID] = stream;
  366. }
  367. for (int i = 0; i < streamCountSecondaryAudio; i++)
  368. {
  369. var stream = CreatePlaylistStream(data, ref pos);
  370. if (stream != null) PlaylistStreams[stream.PID] = stream;
  371. pos += 2;
  372. }
  373. for (int i = 0; i < streamCountSecondaryVideo; i++)
  374. {
  375. var stream = CreatePlaylistStream(data, ref pos);
  376. if (stream != null) PlaylistStreams[stream.PID] = stream;
  377. pos += 6;
  378. }
  379. /*
  380. * TODO
  381. *
  382. for (int i = 0; i < streamCountPIP; i++)
  383. {
  384. TSStream stream = CreatePlaylistStream(data, ref pos);
  385. if (stream != null) PlaylistStreams[stream.PID] = stream;
  386. }
  387. */
  388. pos += itemLength - (pos - itemStart) + 2;
  389. }
  390. pos = chaptersOffset + 4;
  391. int chapterCount = ReadInt16(data, ref pos);
  392. for (int chapterIndex = 0;
  393. chapterIndex < chapterCount;
  394. chapterIndex++)
  395. {
  396. int chapterType = data[pos + 1];
  397. if (chapterType == 1)
  398. {
  399. int streamFileIndex =
  400. ((int)data[pos + 2] << 8) + data[pos + 3];
  401. long chapterTime =
  402. ((long)data[pos + 4] << 24) +
  403. ((long)data[pos + 5] << 16) +
  404. ((long)data[pos + 6] << 8) +
  405. ((long)data[pos + 7]);
  406. var streamClip = chapterClips[streamFileIndex];
  407. double chapterSeconds = (double)chapterTime / 45000;
  408. double relativeSeconds =
  409. chapterSeconds -
  410. streamClip.TimeIn +
  411. streamClip.RelativeTimeIn;
  412. // TODO: Ignore short last chapter?
  413. if (TotalLength - relativeSeconds > 1.0)
  414. {
  415. streamClip.Chapters.Add(chapterSeconds);
  416. this.Chapters.Add(relativeSeconds);
  417. }
  418. }
  419. else
  420. {
  421. // TODO: Handle other chapter types?
  422. }
  423. pos += 14;
  424. }
  425. }
  426. finally
  427. {
  428. if (fileReader != null)
  429. {
  430. fileReader.Dispose();
  431. }
  432. if (fileStream != null)
  433. {
  434. fileStream.Dispose();
  435. }
  436. }
  437. }
  438. public void Initialize()
  439. {
  440. LoadStreamClips();
  441. var clipTimes = new Dictionary<string, List<double>>();
  442. foreach (var clip in StreamClips)
  443. {
  444. if (clip.AngleIndex == 0)
  445. {
  446. if (clipTimes.ContainsKey(clip.Name))
  447. {
  448. if (clipTimes[clip.Name].Contains(clip.TimeIn))
  449. {
  450. HasLoops = true;
  451. break;
  452. }
  453. else
  454. {
  455. clipTimes[clip.Name].Add(clip.TimeIn);
  456. }
  457. }
  458. else
  459. {
  460. clipTimes[clip.Name] = new List<double> { clip.TimeIn };
  461. }
  462. }
  463. }
  464. ClearBitrates();
  465. IsInitialized = true;
  466. }
  467. protected TSStream CreatePlaylistStream(byte[] data, ref int pos)
  468. {
  469. TSStream stream = null;
  470. int start = pos;
  471. int headerLength = data[pos++];
  472. int headerPos = pos;
  473. int headerType = data[pos++];
  474. int pid = 0;
  475. int subpathid = 0;
  476. int subclipid = 0;
  477. switch (headerType)
  478. {
  479. case 1:
  480. pid = ReadInt16(data, ref pos);
  481. break;
  482. case 2:
  483. subpathid = data[pos++];
  484. subclipid = data[pos++];
  485. pid = ReadInt16(data, ref pos);
  486. break;
  487. case 3:
  488. subpathid = data[pos++];
  489. pid = ReadInt16(data, ref pos);
  490. break;
  491. case 4:
  492. subpathid = data[pos++];
  493. subclipid = data[pos++];
  494. pid = ReadInt16(data, ref pos);
  495. break;
  496. default:
  497. break;
  498. }
  499. pos = headerPos + headerLength;
  500. int streamLength = data[pos++];
  501. int streamPos = pos;
  502. var streamType = (TSStreamType)data[pos++];
  503. switch (streamType)
  504. {
  505. case TSStreamType.MVC_VIDEO:
  506. // TODO
  507. break;
  508. case TSStreamType.AVC_VIDEO:
  509. case TSStreamType.MPEG1_VIDEO:
  510. case TSStreamType.MPEG2_VIDEO:
  511. case TSStreamType.VC1_VIDEO:
  512. var videoFormat = (TSVideoFormat)
  513. (data[pos] >> 4);
  514. var frameRate = (TSFrameRate)
  515. (data[pos] & 0xF);
  516. var aspectRatio = (TSAspectRatio)
  517. (data[pos + 1] >> 4);
  518. stream = new TSVideoStream();
  519. ((TSVideoStream)stream).VideoFormat = videoFormat;
  520. ((TSVideoStream)stream).AspectRatio = aspectRatio;
  521. ((TSVideoStream)stream).FrameRate = frameRate;
  522. #if DEBUG
  523. Debug.WriteLine(string.Format(
  524. "\t{0} {1} {2} {3} {4}",
  525. pid,
  526. streamType,
  527. videoFormat,
  528. frameRate,
  529. aspectRatio));
  530. #endif
  531. break;
  532. case TSStreamType.AC3_AUDIO:
  533. case TSStreamType.AC3_PLUS_AUDIO:
  534. case TSStreamType.AC3_PLUS_SECONDARY_AUDIO:
  535. case TSStreamType.AC3_TRUE_HD_AUDIO:
  536. case TSStreamType.DTS_AUDIO:
  537. case TSStreamType.DTS_HD_AUDIO:
  538. case TSStreamType.DTS_HD_MASTER_AUDIO:
  539. case TSStreamType.DTS_HD_SECONDARY_AUDIO:
  540. case TSStreamType.LPCM_AUDIO:
  541. case TSStreamType.MPEG1_AUDIO:
  542. case TSStreamType.MPEG2_AUDIO:
  543. int audioFormat = ReadByte(data, ref pos);
  544. var channelLayout = (TSChannelLayout)
  545. (audioFormat >> 4);
  546. var sampleRate = (TSSampleRate)
  547. (audioFormat & 0xF);
  548. string audioLanguage = ReadString(data, 3, ref pos);
  549. stream = new TSAudioStream();
  550. ((TSAudioStream)stream).ChannelLayout = channelLayout;
  551. ((TSAudioStream)stream).SampleRate = TSAudioStream.ConvertSampleRate(sampleRate);
  552. ((TSAudioStream)stream).LanguageCode = audioLanguage;
  553. #if DEBUG
  554. Debug.WriteLine(string.Format(
  555. "\t{0} {1} {2} {3} {4}",
  556. pid,
  557. streamType,
  558. audioLanguage,
  559. channelLayout,
  560. sampleRate));
  561. #endif
  562. break;
  563. case TSStreamType.INTERACTIVE_GRAPHICS:
  564. case TSStreamType.PRESENTATION_GRAPHICS:
  565. string graphicsLanguage = ReadString(data, 3, ref pos);
  566. stream = new TSGraphicsStream();
  567. ((TSGraphicsStream)stream).LanguageCode = graphicsLanguage;
  568. if (data[pos] != 0)
  569. {
  570. }
  571. #if DEBUG
  572. Debug.WriteLine(string.Format(
  573. "\t{0} {1} {2}",
  574. pid,
  575. streamType,
  576. graphicsLanguage));
  577. #endif
  578. break;
  579. case TSStreamType.SUBTITLE:
  580. int code = ReadByte(data, ref pos); // TODO
  581. string textLanguage = ReadString(data, 3, ref pos);
  582. stream = new TSTextStream();
  583. ((TSTextStream)stream).LanguageCode = textLanguage;
  584. #if DEBUG
  585. Debug.WriteLine(string.Format(
  586. "\t{0} {1} {2}",
  587. pid,
  588. streamType,
  589. textLanguage));
  590. #endif
  591. break;
  592. default:
  593. break;
  594. }
  595. pos = streamPos + streamLength;
  596. if (stream != null)
  597. {
  598. stream.PID = (ushort)pid;
  599. stream.StreamType = streamType;
  600. }
  601. return stream;
  602. }
  603. private void LoadStreamClips()
  604. {
  605. AngleClips.Clear();
  606. if (AngleCount > 0)
  607. {
  608. for (int angleIndex = 0; angleIndex < AngleCount; angleIndex++)
  609. {
  610. AngleClips.Add(new Dictionary<double, TSStreamClip>());
  611. }
  612. }
  613. TSStreamClip referenceClip = null;
  614. if (StreamClips.Count > 0)
  615. {
  616. referenceClip = StreamClips[0];
  617. }
  618. foreach (var clip in StreamClips)
  619. {
  620. if (clip.StreamClipFile.Streams.Count > referenceClip.StreamClipFile.Streams.Count)
  621. {
  622. referenceClip = clip;
  623. }
  624. else if (clip.Length > referenceClip.Length)
  625. {
  626. referenceClip = clip;
  627. }
  628. if (AngleCount > 0)
  629. {
  630. if (clip.AngleIndex == 0)
  631. {
  632. for (int angleIndex = 0; angleIndex < AngleCount; angleIndex++)
  633. {
  634. AngleClips[angleIndex][clip.RelativeTimeIn] = clip;
  635. }
  636. }
  637. else
  638. {
  639. AngleClips[clip.AngleIndex - 1][clip.RelativeTimeIn] = clip;
  640. }
  641. }
  642. }
  643. foreach (var clipStream
  644. in referenceClip.StreamClipFile.Streams.Values)
  645. {
  646. if (!Streams.ContainsKey(clipStream.PID))
  647. {
  648. var stream = clipStream.Clone();
  649. Streams[clipStream.PID] = stream;
  650. if (!IsCustom && !PlaylistStreams.ContainsKey(stream.PID))
  651. {
  652. stream.IsHidden = true;
  653. HasHiddenTracks = true;
  654. }
  655. if (stream.IsVideoStream)
  656. {
  657. VideoStreams.Add((TSVideoStream)stream);
  658. }
  659. else if (stream.IsAudioStream)
  660. {
  661. AudioStreams.Add((TSAudioStream)stream);
  662. }
  663. else if (stream.IsGraphicsStream)
  664. {
  665. GraphicsStreams.Add((TSGraphicsStream)stream);
  666. }
  667. else if (stream.IsTextStream)
  668. {
  669. TextStreams.Add((TSTextStream)stream);
  670. }
  671. }
  672. }
  673. if (referenceClip.StreamFile != null)
  674. {
  675. // TODO: Better way to add this in?
  676. if (BDInfoSettings.EnableSSIF &&
  677. referenceClip.StreamFile.InterleavedFile != null &&
  678. referenceClip.StreamFile.Streams.ContainsKey(4114) &&
  679. !Streams.ContainsKey(4114))
  680. {
  681. var stream = referenceClip.StreamFile.Streams[4114].Clone();
  682. Streams[4114] = stream;
  683. if (stream.IsVideoStream)
  684. {
  685. VideoStreams.Add((TSVideoStream)stream);
  686. }
  687. }
  688. foreach (var clipStream
  689. in referenceClip.StreamFile.Streams.Values)
  690. {
  691. if (Streams.ContainsKey(clipStream.PID))
  692. {
  693. var stream = Streams[clipStream.PID];
  694. if (stream.StreamType != clipStream.StreamType) continue;
  695. if (clipStream.BitRate > stream.BitRate)
  696. {
  697. stream.BitRate = clipStream.BitRate;
  698. }
  699. stream.IsVBR = clipStream.IsVBR;
  700. if (stream.IsVideoStream &&
  701. clipStream.IsVideoStream)
  702. {
  703. ((TSVideoStream)stream).EncodingProfile =
  704. ((TSVideoStream)clipStream).EncodingProfile;
  705. }
  706. else if (stream.IsAudioStream &&
  707. clipStream.IsAudioStream)
  708. {
  709. var audioStream = (TSAudioStream)stream;
  710. var clipAudioStream = (TSAudioStream)clipStream;
  711. if (clipAudioStream.ChannelCount > audioStream.ChannelCount)
  712. {
  713. audioStream.ChannelCount = clipAudioStream.ChannelCount;
  714. }
  715. if (clipAudioStream.LFE > audioStream.LFE)
  716. {
  717. audioStream.LFE = clipAudioStream.LFE;
  718. }
  719. if (clipAudioStream.SampleRate > audioStream.SampleRate)
  720. {
  721. audioStream.SampleRate = clipAudioStream.SampleRate;
  722. }
  723. if (clipAudioStream.BitDepth > audioStream.BitDepth)
  724. {
  725. audioStream.BitDepth = clipAudioStream.BitDepth;
  726. }
  727. if (clipAudioStream.DialNorm < audioStream.DialNorm)
  728. {
  729. audioStream.DialNorm = clipAudioStream.DialNorm;
  730. }
  731. if (clipAudioStream.AudioMode != TSAudioMode.Unknown)
  732. {
  733. audioStream.AudioMode = clipAudioStream.AudioMode;
  734. }
  735. if (clipAudioStream.CoreStream != null &&
  736. audioStream.CoreStream == null)
  737. {
  738. audioStream.CoreStream = (TSAudioStream)
  739. clipAudioStream.CoreStream.Clone();
  740. }
  741. }
  742. }
  743. }
  744. }
  745. for (int i = 0; i < AngleCount; i++)
  746. {
  747. AngleStreams.Add(new Dictionary<ushort, TSStream>());
  748. }
  749. if (!BDInfoSettings.KeepStreamOrder)
  750. {
  751. VideoStreams.Sort(CompareVideoStreams);
  752. }
  753. foreach (TSStream stream in VideoStreams)
  754. {
  755. SortedStreams.Add(stream);
  756. for (int i = 0; i < AngleCount; i++)
  757. {
  758. var angleStream = stream.Clone();
  759. angleStream.AngleIndex = i + 1;
  760. AngleStreams[i][angleStream.PID] = angleStream;
  761. SortedStreams.Add(angleStream);
  762. }
  763. }
  764. if (!BDInfoSettings.KeepStreamOrder)
  765. {
  766. AudioStreams.Sort(CompareAudioStreams);
  767. }
  768. foreach (TSStream stream in AudioStreams)
  769. {
  770. SortedStreams.Add(stream);
  771. }
  772. if (!BDInfoSettings.KeepStreamOrder)
  773. {
  774. GraphicsStreams.Sort(CompareGraphicsStreams);
  775. }
  776. foreach (TSStream stream in GraphicsStreams)
  777. {
  778. SortedStreams.Add(stream);
  779. }
  780. if (!BDInfoSettings.KeepStreamOrder)
  781. {
  782. TextStreams.Sort(CompareTextStreams);
  783. }
  784. foreach (TSStream stream in TextStreams)
  785. {
  786. SortedStreams.Add(stream);
  787. }
  788. }
  789. public void ClearBitrates()
  790. {
  791. foreach (var clip in StreamClips)
  792. {
  793. clip.PayloadBytes = 0;
  794. clip.PacketCount = 0;
  795. clip.PacketSeconds = 0;
  796. if (clip.StreamFile != null)
  797. {
  798. foreach (var stream in clip.StreamFile.Streams.Values)
  799. {
  800. stream.PayloadBytes = 0;
  801. stream.PacketCount = 0;
  802. stream.PacketSeconds = 0;
  803. }
  804. if (clip.StreamFile != null &&
  805. clip.StreamFile.StreamDiagnostics != null)
  806. {
  807. clip.StreamFile.StreamDiagnostics.Clear();
  808. }
  809. }
  810. }
  811. foreach (var stream in SortedStreams)
  812. {
  813. stream.PayloadBytes = 0;
  814. stream.PacketCount = 0;
  815. stream.PacketSeconds = 0;
  816. }
  817. }
  818. public bool IsValid
  819. {
  820. get
  821. {
  822. if (!IsInitialized) return false;
  823. if (BDInfoSettings.FilterShortPlaylists &&
  824. TotalLength < BDInfoSettings.FilterShortPlaylistsValue)
  825. {
  826. return false;
  827. }
  828. if (HasLoops &&
  829. BDInfoSettings.FilterLoopingPlaylists)
  830. {
  831. return false;
  832. }
  833. return true;
  834. }
  835. }
  836. public int CompareVideoStreams(
  837. TSVideoStream x,
  838. TSVideoStream y)
  839. {
  840. if (x == null && y == null)
  841. {
  842. return 0;
  843. }
  844. else if (x == null && y != null)
  845. {
  846. return 1;
  847. }
  848. else if (x != null && y == null)
  849. {
  850. return -1;
  851. }
  852. else
  853. {
  854. if (x.Height > y.Height)
  855. {
  856. return -1;
  857. }
  858. else if (y.Height > x.Height)
  859. {
  860. return 1;
  861. }
  862. else if (x.PID > y.PID)
  863. {
  864. return 1;
  865. }
  866. else if (y.PID > x.PID)
  867. {
  868. return -1;
  869. }
  870. else
  871. {
  872. return 0;
  873. }
  874. }
  875. }
  876. public int CompareAudioStreams(
  877. TSAudioStream x,
  878. TSAudioStream y)
  879. {
  880. if (x == y)
  881. {
  882. return 0;
  883. }
  884. else if (x == null && y == null)
  885. {
  886. return 0;
  887. }
  888. else if (x == null && y != null)
  889. {
  890. return -1;
  891. }
  892. else if (x != null && y == null)
  893. {
  894. return 1;
  895. }
  896. else
  897. {
  898. if (x.ChannelCount > y.ChannelCount)
  899. {
  900. return -1;
  901. }
  902. else if (y.ChannelCount > x.ChannelCount)
  903. {
  904. return 1;
  905. }
  906. else
  907. {
  908. int sortX = GetStreamTypeSortIndex(x.StreamType);
  909. int sortY = GetStreamTypeSortIndex(y.StreamType);
  910. if (sortX > sortY)
  911. {
  912. return -1;
  913. }
  914. else if (sortY > sortX)
  915. {
  916. return 1;
  917. }
  918. else
  919. {
  920. if (x.LanguageCode == "eng")
  921. {
  922. return -1;
  923. }
  924. else if (y.LanguageCode == "eng")
  925. {
  926. return 1;
  927. }
  928. else if (x.LanguageCode != y.LanguageCode)
  929. {
  930. return string.Compare(
  931. x.LanguageName, y.LanguageName);
  932. }
  933. else if (x.PID < y.PID)
  934. {
  935. return -1;
  936. }
  937. else if (y.PID < x.PID)
  938. {
  939. return 1;
  940. }
  941. return 0;
  942. }
  943. }
  944. }
  945. }
  946. public int CompareTextStreams(
  947. TSTextStream x,
  948. TSTextStream y)
  949. {
  950. if (x == y)
  951. {
  952. return 0;
  953. }
  954. else if (x == null && y == null)
  955. {
  956. return 0;
  957. }
  958. else if (x == null && y != null)
  959. {
  960. return -1;
  961. }
  962. else if (x != null && y == null)
  963. {
  964. return 1;
  965. }
  966. else
  967. {
  968. if (x.LanguageCode == "eng")
  969. {
  970. return -1;
  971. }
  972. else if (y.LanguageCode == "eng")
  973. {
  974. return 1;
  975. }
  976. else
  977. {
  978. if (x.LanguageCode == y.LanguageCode)
  979. {
  980. if (x.PID > y.PID)
  981. {
  982. return 1;
  983. }
  984. else if (y.PID > x.PID)
  985. {
  986. return -1;
  987. }
  988. else
  989. {
  990. return 0;
  991. }
  992. }
  993. else
  994. {
  995. return string.Compare(
  996. x.LanguageName, y.LanguageName);
  997. }
  998. }
  999. }
  1000. }
  1001. private int CompareGraphicsStreams(
  1002. TSGraphicsStream x,
  1003. TSGraphicsStream y)
  1004. {
  1005. if (x == y)
  1006. {
  1007. return 0;
  1008. }
  1009. else if (x == null && y == null)
  1010. {
  1011. return 0;
  1012. }
  1013. else if (x == null && y != null)
  1014. {
  1015. return -1;
  1016. }
  1017. else if (x != null && y == null)
  1018. {
  1019. return 1;
  1020. }
  1021. else
  1022. {
  1023. int sortX = GetStreamTypeSortIndex(x.StreamType);
  1024. int sortY = GetStreamTypeSortIndex(y.StreamType);
  1025. if (sortX > sortY)
  1026. {
  1027. return -1;
  1028. }
  1029. else if (sortY > sortX)
  1030. {
  1031. return 1;
  1032. }
  1033. else if (x.LanguageCode == "eng")
  1034. {
  1035. return -1;
  1036. }
  1037. else if (y.LanguageCode == "eng")
  1038. {
  1039. return 1;
  1040. }
  1041. else
  1042. {
  1043. if (x.LanguageCode == y.LanguageCode)
  1044. {
  1045. if (x.PID > y.PID)
  1046. {
  1047. return 1;
  1048. }
  1049. else if (y.PID > x.PID)
  1050. {
  1051. return -1;
  1052. }
  1053. else
  1054. {
  1055. return 0;
  1056. }
  1057. }
  1058. else
  1059. {
  1060. return string.Compare(x.LanguageName, y.LanguageName);
  1061. }
  1062. }
  1063. }
  1064. }
  1065. private int GetStreamTypeSortIndex(TSStreamType streamType)
  1066. {
  1067. switch (streamType)
  1068. {
  1069. case TSStreamType.Unknown:
  1070. return 0;
  1071. case TSStreamType.MPEG1_VIDEO:
  1072. return 1;
  1073. case TSStreamType.MPEG2_VIDEO:
  1074. return 2;
  1075. case TSStreamType.AVC_VIDEO:
  1076. return 3;
  1077. case TSStreamType.VC1_VIDEO:
  1078. return 4;
  1079. case TSStreamType.MVC_VIDEO:
  1080. return 5;
  1081. case TSStreamType.MPEG1_AUDIO:
  1082. return 1;
  1083. case TSStreamType.MPEG2_AUDIO:
  1084. return 2;
  1085. case TSStreamType.AC3_PLUS_SECONDARY_AUDIO:
  1086. return 3;
  1087. case TSStreamType.DTS_HD_SECONDARY_AUDIO:
  1088. return 4;
  1089. case TSStreamType.AC3_AUDIO:
  1090. return 5;
  1091. case TSStreamType.DTS_AUDIO:
  1092. return 6;
  1093. case TSStreamType.AC3_PLUS_AUDIO:
  1094. return 7;
  1095. case TSStreamType.DTS_HD_AUDIO:
  1096. return 8;
  1097. case TSStreamType.AC3_TRUE_HD_AUDIO:
  1098. return 9;
  1099. case TSStreamType.DTS_HD_MASTER_AUDIO:
  1100. return 10;
  1101. case TSStreamType.LPCM_AUDIO:
  1102. return 11;
  1103. case TSStreamType.SUBTITLE:
  1104. return 1;
  1105. case TSStreamType.INTERACTIVE_GRAPHICS:
  1106. return 2;
  1107. case TSStreamType.PRESENTATION_GRAPHICS:
  1108. return 3;
  1109. default:
  1110. return 0;
  1111. }
  1112. }
  1113. protected string ReadString(
  1114. byte[] data,
  1115. int count,
  1116. ref int pos)
  1117. {
  1118. string val = Encoding.ASCII.GetString(data, pos, count);
  1119. pos += count;
  1120. return val;
  1121. }
  1122. protected int ReadInt32(
  1123. byte[] data,
  1124. ref int pos)
  1125. {
  1126. int val =
  1127. ((int)data[pos] << 24) +
  1128. ((int)data[pos + 1] << 16) +
  1129. ((int)data[pos + 2] << 8) +
  1130. ((int)data[pos + 3]);
  1131. pos += 4;
  1132. return val;
  1133. }
  1134. protected int ReadInt16(
  1135. byte[] data,
  1136. ref int pos)
  1137. {
  1138. int val =
  1139. ((int)data[pos] << 8) +
  1140. ((int)data[pos + 1]);
  1141. pos += 2;
  1142. return val;
  1143. }
  1144. protected byte ReadByte(
  1145. byte[] data,
  1146. ref int pos)
  1147. {
  1148. return data[pos++];
  1149. }
  1150. }
  1151. }