VirtualBox

source: vbox/trunk/src/VBox/Storage/CUE.cpp@ 66280

Last change on this file since 66280 was 66211, checked in by vboxsync, 8 years ago

vdmedia.h,++: Rename VDTYPE_DVD to VDTYPE_OPTICAL_DISC

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 68.2 KB
Line 
1/* $Id: CUE.cpp 66211 2017-03-22 19:44:19Z vboxsync $ */
2/** @file
3 * CUE - CUE/BIN Disk image, Core Code.
4 */
5
6/*
7 * Copyright (C) 2017 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 */
17
18
19/*********************************************************************************************************************************
20* Header Files *
21*********************************************************************************************************************************/
22#define LOG_GROUP LOG_GROUP_VD_CUE
23#include <VBox/vd-plugin.h>
24#include <VBox/err.h>
25
26#include <VBox/log.h>
27#include <VBox/scsiinline.h>
28#include <iprt/assert.h>
29#include <iprt/asm.h>
30#include <iprt/alloc.h>
31#include <iprt/cdefs.h>
32#include <iprt/ctype.h>
33#include <iprt/path.h>
34#include <iprt/string.h>
35
36#include "VDBackends.h"
37
38
39/*********************************************************************************************************************************
40* Constants And Macros, Structures and Typedefs *
41*********************************************************************************************************************************/
42
43/**
44 * CUE descriptor file token type.
45 */
46typedef enum CUETOKENTYPE
47{
48 /** Invalid token type. */
49 CUETOKENTYPE_INVALID = 0,
50 /** Reserved keyword. */
51 CUETOKENTYPE_KEYWORD,
52 /** String token. */
53 CUETOKENTYPE_STRING,
54 /** Unsigned integer. */
55 CUETOKENTYPE_INTEGER_UNSIGNED,
56 /** MSF (mm:ss:ff) location token. */
57 CUETOKENTYPE_MSF,
58 /** Error token (unexpected character found). */
59 CUETOKENTYPE_ERROR,
60 /** End of stream token. */
61 CUETOKENTYPE_EOS
62} CUETOKENTYPE;
63
64/**
65 * CUE reservered keyword type.
66 */
67typedef enum CUEKEYWORD
68{
69 /** Invalid keyword. */
70 CUEKEYWORD_INVALID = 0,
71 /** FILE. */
72 CUEKEYWORD_FILE,
73 /** BINARY */
74 CUEKEYWORD_BINARY,
75 /** MOTOROLA */
76 CUEKEYWORD_MOTOROLA,
77 /** WAVE */
78 CUEKEYWORD_WAVE,
79 /** MP3 */
80 CUEKEYWORD_MP3,
81 /** AIFF */
82 CUEKEYWORD_AIFF,
83 /** CATALOG */
84 CUEKEYWORD_CATALOG,
85 CUEKEYWORD_CDTEXTFILE,
86 CUEKEYWORD_FLAGS,
87 CUEKEYWORD_INDEX,
88 CUEKEYWORD_ISRC,
89 CUEKEYWORD_PERFORMER,
90 CUEKEYWORD_POSTGAP,
91 CUEKEYWORD_PREGAP,
92 CUEKEYWORD_SONGWRITER,
93 CUEKEYWORD_TITLE,
94 CUEKEYWORD_TRACK,
95 CUEKEYWORD_MODE1_2048,
96 CUEKEYWORD_MODE1_2352,
97 CUEKEYWORD_AUDIO,
98 CUEKEYWORD_REM
99} CUEKEYWORD;
100
101/**
102 * CUE sheet token.
103 */
104typedef struct CUETOKEN
105{
106 /** The token type. */
107 CUETOKENTYPE enmType;
108 /** Token type dependent data. */
109 union
110 {
111 /** Keyword token. */
112 struct
113 {
114 /** The keyword enumerator. */
115 CUEKEYWORD enmKeyword;
116 } Keyword;
117 /** String token (without quotation marks). */
118 struct
119 {
120 /** Pointer to the start of the string. */
121 const char *psz;
122 /** Number of characters for the string excluding the null terminator. */
123 size_t cch;
124 } String;
125 /** Integer token. */
126 struct
127 {
128 /** Numerical constant. */
129 uint64_t u64;
130 } Int;
131 /** MSF location token. */
132 struct
133 {
134 /** Minute part. */
135 uint8_t u8Minute;
136 /** Second part. */
137 uint8_t u8Second;
138 /** Frame part. */
139 uint8_t u8Frame;
140 } Msf;
141 } Type;
142} CUETOKEN;
143/** Pointer to a CUE sheet token. */
144typedef CUETOKEN *PCUETOKEN;
145/** Pointer to a const CUE sheet token. */
146typedef const CUETOKEN *PCCUETOKEN;
147
148/**
149 * CUE tokenizer state.
150 */
151typedef struct CUETOKENIZER
152{
153 /** Char buffer to read from. */
154 const char *pszInput;
155 /** Token 1. */
156 CUETOKEN Token1;
157 /** Token 2. */
158 CUETOKEN Token2;
159 /** Pointer to the current active token. */
160 PCUETOKEN pTokenCurr;
161 /** The next token in the input stream (used for peeking). */
162 PCUETOKEN pTokenNext;
163} CUETOKENIZER;
164/** Pointer to a CUE tokenizer state. */
165typedef CUETOKENIZER *PCUETOKENIZER;
166
167/**
168 * CUE keyword entry.
169 */
170typedef struct CUEKEYWORDDESC
171{
172 /** Keyword string. */
173 const char *pszKeyword;
174 /** Size of the string in characters without zero terminator. */
175 size_t cchKeyword;
176 /** Keyword type. */
177 CUEKEYWORD enmKeyword;
178} CUEKEYWORDDESC;
179/** Pointer to a CUE keyword entry. */
180typedef CUEKEYWORDDESC *PCUEKEYWORDDESC;
181/** Pointer to a const CUE keyword entry. */
182typedef const CUEKEYWORDDESC *PCCUEKEYWORDDESC;
183
184/**
185 * CUE image data structure.
186 */
187typedef struct CUEIMAGE
188{
189 /** Image name. */
190 const char *pszFilename;
191 /** Storage handle. */
192 PVDIOSTORAGE pStorage;
193 /** The backing file containing the actual data. */
194 char *pszDataFilename;
195 /** Storage handle for the backing file. */
196 PVDIOSTORAGE pStorageData;
197
198 /** Pointer to the per-disk VD interface list. */
199 PVDINTERFACE pVDIfsDisk;
200 /** Pointer to the per-image VD interface list. */
201 PVDINTERFACE pVDIfsImage;
202 /** Error interface. */
203 PVDINTERFACEERROR pIfError;
204 /** I/O interface. */
205 PVDINTERFACEIOINT pIfIo;
206
207 /** Open flags passed by VD layer. */
208 unsigned uOpenFlags;
209 /** Image flags defined during creation or determined during open. */
210 unsigned uImageFlags;
211 /** Maximum number of tracks the region list can hold. */
212 uint32_t cTracksMax;
213 /** Pointer to our internal region list. */
214 PVDREGIONLIST pRegionList;
215 /** Flag whether the backing file is little (BINARY) or big (MOTOROLA) endian. */
216 bool fLittleEndian;
217} CUEIMAGE, *PCUEIMAGE;
218
219
220/*********************************************************************************************************************************
221* Static Variables *
222*********************************************************************************************************************************/
223
224/** NULL-terminated array of supported file extensions. */
225static const VDFILEEXTENSION s_aCueFileExtensions[] =
226{
227 {"cue", VDTYPE_OPTICAL_DISC},
228 {NULL, VDTYPE_INVALID}
229};
230
231/**
232 * Known keywords.
233 */
234static const CUEKEYWORDDESC g_aCueKeywords[] =
235{
236 {RT_STR_TUPLE("FILE"), CUEKEYWORD_FILE},
237 {RT_STR_TUPLE("BINARY"), CUEKEYWORD_BINARY},
238 {RT_STR_TUPLE("WAVE"), CUEKEYWORD_WAVE},
239 {RT_STR_TUPLE("MP3"), CUEKEYWORD_MP3},
240 {RT_STR_TUPLE("AIFF"), CUEKEYWORD_AIFF},
241 {RT_STR_TUPLE("CATALOG"), CUEKEYWORD_CATALOG},
242 {RT_STR_TUPLE("CDTEXTFILE"), CUEKEYWORD_CDTEXTFILE},
243 {RT_STR_TUPLE("FLAGS"), CUEKEYWORD_FLAGS},
244 {RT_STR_TUPLE("INDEX"), CUEKEYWORD_INDEX},
245 {RT_STR_TUPLE("ISRC"), CUEKEYWORD_ISRC},
246 {RT_STR_TUPLE("PERFORMER"), CUEKEYWORD_PERFORMER},
247 {RT_STR_TUPLE("POSTGAP"), CUEKEYWORD_POSTGAP},
248 {RT_STR_TUPLE("PREGAP"), CUEKEYWORD_PREGAP},
249 {RT_STR_TUPLE("SONGWRITER"), CUEKEYWORD_SONGWRITER},
250 {RT_STR_TUPLE("TITLE"), CUEKEYWORD_TITLE},
251 {RT_STR_TUPLE("TRACK"), CUEKEYWORD_TRACK},
252 {RT_STR_TUPLE("MODE1/2048"), CUEKEYWORD_MODE1_2048},
253 {RT_STR_TUPLE("MODE1/2352"), CUEKEYWORD_MODE1_2352},
254 {RT_STR_TUPLE("AUDIO"), CUEKEYWORD_AUDIO},
255 {RT_STR_TUPLE("REM"), CUEKEYWORD_REM}
256};
257
258
259/*********************************************************************************************************************************
260* Internal Functions *
261*********************************************************************************************************************************/
262
263/**
264 * Ensures that the region list can hold up to the given number of tracks.
265 *
266 * @returns VBox status code.
267 * @param pThis The CUE image state.
268 * @param cTracksMax Maximum number of tracks.
269 */
270static int cueEnsureRegionListSize(PCUEIMAGE pThis, uint32_t cTracksMax)
271{
272 int rc = VINF_SUCCESS;
273
274 if (pThis->cTracksMax < cTracksMax)
275 {
276 PVDREGIONLIST pRegionListNew = (PVDREGIONLIST)RTMemRealloc(pThis->pRegionList,
277 RT_UOFFSETOF(VDREGIONLIST, aRegions[cTracksMax]));
278 if (pRegionListNew)
279 {
280 /* Init all the new allocated tracks. */
281 for (uint32_t i = pThis->cTracksMax; i < cTracksMax; i++)
282 pRegionListNew->aRegions[i].offRegion = UINT64_MAX;
283 pThis->pRegionList = pRegionListNew;
284 pThis->cTracksMax = cTracksMax;
285 }
286 else
287 rc = VERR_NO_MEMORY;
288 }
289
290 return rc;
291}
292
293/**
294 * Returns whether the tokenizer reached the end of the stream.
295 *
296 * @returns true if the tokenizer reached the end of stream marker
297 * false otherwise.
298 * @param pTokenizer The tokenizer state.
299 */
300DECLINLINE(bool) cueTokenizerIsEos(PCUETOKENIZER pTokenizer)
301{
302 return *pTokenizer->pszInput == '\0';
303}
304
305/**
306 * Skip one character in the input stream.
307 *
308 * @returns nothing.
309 * @param pTokenizer The tokenizer state.
310 */
311DECLINLINE(void) cueTokenizerSkipCh(PCUETOKENIZER pTokenizer)
312{
313 pTokenizer->pszInput++;
314}
315
316/**
317 * Returns the next char in the input buffer without advancing it.
318 *
319 * @returns Next character in the input buffer.
320 * @param pTokenizer The tokenizer state.
321 */
322DECLINLINE(char) cueTokenizerPeekCh(PCUETOKENIZER pTokenizer)
323{
324 return cueTokenizerIsEos(pTokenizer)
325 ? '\0'
326 : *(pTokenizer->pszInput + 1);
327}
328
329/**
330 * Returns the next character in the input buffer advancing the internal
331 * position.
332 *
333 * @returns Next character in the stream.
334 * @param pTokenizer The tokenizer state.
335 */
336DECLINLINE(char) cueTokenizerGetCh(PCUETOKENIZER pTokenizer)
337{
338 char ch;
339
340 if (cueTokenizerIsEos(pTokenizer))
341 ch = '\0';
342 else
343 ch = *pTokenizer->pszInput;
344
345 return ch;
346}
347
348/**
349 * Sets a new line for the tokenizer.
350 *
351 * @returns nothing.
352 * @param pTokenizer The tokenizer state.
353 * @param cSkip How many characters to skip.
354 */
355DECLINLINE(void) cueTokenizerNewLine(PCUETOKENIZER pTokenizer, unsigned cSkip)
356{
357 pTokenizer->pszInput += cSkip;
358}
359
360/**
361 * Checks whether the current position in the input stream is a new line
362 * and skips it.
363 *
364 * @returns Flag whether there was a new line at the current position
365 * in the input buffer.
366 * @param pTokenizer The tokenizer state.
367 */
368DECLINLINE(bool) cueTokenizerIsSkipNewLine(PCUETOKENIZER pTokenizer)
369{
370 bool fNewline = true;
371
372 if ( cueTokenizerGetCh(pTokenizer) == '\r'
373 && cueTokenizerPeekCh(pTokenizer) == '\n')
374 cueTokenizerNewLine(pTokenizer, 2);
375 else if (cueTokenizerGetCh(pTokenizer) == '\n')
376 cueTokenizerNewLine(pTokenizer, 1);
377 else
378 fNewline = false;
379
380 return fNewline;
381}
382
383/**
384 * Skips a multi line comment.
385 *
386 * @returns nothing.
387 * @param pTokenizer The tokenizer state.
388 */
389DECLINLINE(void) cueTokenizerSkipComment(PCUETOKENIZER pTokenizer)
390{
391 while ( !cueTokenizerIsEos(pTokenizer)
392 && !cueTokenizerIsSkipNewLine(pTokenizer))
393 cueTokenizerSkipCh(pTokenizer);
394}
395
396/**
397 * Skip all whitespace starting from the current input buffer position.
398 * Skips all present comments too.
399 *
400 * @returns nothing.
401 * @param pTokenizer The tokenizer state.
402 */
403DECLINLINE(void) cueTokenizerSkipWhitespace(PCUETOKENIZER pTokenizer)
404{
405 while (!cueTokenizerIsEos(pTokenizer))
406 {
407 while ( cueTokenizerGetCh(pTokenizer) == ' '
408 || cueTokenizerGetCh(pTokenizer) == '\t')
409 cueTokenizerSkipCh(pTokenizer);
410
411 if ( !cueTokenizerIsEos(pTokenizer)
412 && !cueTokenizerIsSkipNewLine(pTokenizer))
413 break; /* Skipped everything, next is some real content. */
414 }
415}
416
417/**
418 * Get an identifier token from the tokenizer.
419 *
420 * @returns nothing.
421 * @param pTokenizer The tokenizer state.
422 * @param pToken The uninitialized token.
423 */
424static void cueTokenizerGetKeyword(PCUETOKENIZER pTokenizer, PCUETOKEN pToken)
425{
426 char ch;
427 unsigned cchKeyword = 0;
428 bool fIsKeyword = false;
429 bool fIsComment = false;
430 const char *pszKeyword = pTokenizer->pszInput;
431
432 Assert(RT_C_IS_ALPHA(*pszKeyword));
433
434 do
435 {
436 fIsComment = false;
437
438 do
439 {
440 cchKeyword++;
441 cueTokenizerSkipCh(pTokenizer);
442 ch = cueTokenizerGetCh(pTokenizer);
443 }
444 while (RT_C_IS_ALNUM(ch) || ch == '_' || ch == '/' || ch == '.');
445
446 /* Check whether we got a keyword or a string constant. */
447 for (unsigned i = 0; i < RT_ELEMENTS(g_aCueKeywords); i++)
448 {
449 if (!RTStrNCmp(g_aCueKeywords[i].pszKeyword, pszKeyword, RT_MIN(cchKeyword, g_aCueKeywords[i].cchKeyword)))
450 {
451 if (g_aCueKeywords[i].enmKeyword == CUEKEYWORD_REM)
452 {
453 /* The REM keyword is handled here as it indicates a comment which we just skip. */
454 cueTokenizerSkipComment(pTokenizer);
455 fIsComment = true;
456 }
457 else
458 {
459 fIsKeyword = true;
460 pToken->enmType = CUETOKENTYPE_KEYWORD;
461 pToken->Type.Keyword.enmKeyword = g_aCueKeywords[i].enmKeyword;
462 }
463 break;
464 }
465 }
466 } while (fIsComment);
467
468 /* Make it a string. */
469 if (ch == '\0')
470 pToken->enmType = CUETOKENTYPE_EOS;
471 else if (!fIsKeyword)
472 {
473 pToken->enmType = CUETOKENTYPE_STRING;
474 pToken->Type.String.psz = pszKeyword;
475 pToken->Type.String.cch = cchKeyword;
476 }
477}
478
479/**
480 * Get an integer value or MSF location indicator from the tokenizer.
481 *
482 * @returns nothing.
483 * @param pTokenizer The tokenizer state.
484 * @param pToken The uninitialized token.
485 */
486static void cueTokenizerGetIntegerOrMsf(PCUETOKENIZER pTokenizer, PCUETOKEN pToken)
487{
488 char szInt[20 + 1]; /* Maximum which fits into an unsigned 64bit integer + zero terminator. */
489 unsigned cchInt = 0;
490 bool fMsf = false;
491 char ch = cueTokenizerGetCh(pTokenizer);
492
493 Assert(RT_C_IS_DIGIT(ch));
494 RT_ZERO(szInt);
495
496 /* Go through the characters and check for the : mark which denotes MSF location indicator. */
497 do
498 {
499 szInt[cchInt++] = ch;
500 cueTokenizerSkipCh(pTokenizer);
501 ch = cueTokenizerGetCh(pTokenizer);
502 if (ch == ':')
503 fMsf = true;
504 }
505 while ( (RT_C_IS_DIGIT(ch) || ch == ':')
506 && cchInt < sizeof(szInt));
507
508 if (cchInt < sizeof(szInt) - 1)
509 {
510 if (fMsf)
511 {
512 /* Check that the format matches our expectations (mm:ss:ff). */
513 if ( cchInt == 8 && szInt[2] == ':' && szInt[5] == ':')
514 {
515 /* Parse the single fields. */
516 szInt[2] = '\0';
517 szInt[5] = '\0';
518
519 int rc = RTStrToUInt8Full(&szInt[0], 10, &pToken->Type.Msf.u8Minute);
520 if (RT_SUCCESS(rc))
521 rc = RTStrToUInt8Full(&szInt[3], 10, &pToken->Type.Msf.u8Second);
522 if (RT_SUCCESS(rc))
523 rc = RTStrToUInt8Full(&szInt[6], 10, &pToken->Type.Msf.u8Frame);
524 if (RT_SUCCESS(rc))
525 pToken->enmType = CUETOKENTYPE_MSF;
526 else
527 pToken->enmType = CUETOKENTYPE_ERROR;
528 }
529 else
530 pToken->enmType = CUETOKENTYPE_ERROR;
531 }
532 else
533 {
534 pToken->enmType = CUETOKENTYPE_INTEGER_UNSIGNED;
535 int rc = RTStrToUInt64Full(&szInt[0], 10, &pToken->Type.Int.u64);
536 if (RT_FAILURE(rc))
537 pToken->enmType = CUETOKENTYPE_ERROR;
538 }
539 }
540 else
541 pToken->enmType = CUETOKENTYPE_ERROR;
542}
543
544/**
545 * Parses a string constant.
546 *
547 * @returns nothing.
548 * @param pTokenizer The tokenizer state.
549 * @param pToken The uninitialized token.
550 *
551 * @remarks: No escape sequences allowed at this time.
552 */
553static void cueTokenizerGetStringConst(PCUETOKENIZER pTokenizer, PCUETOKEN pToken)
554{
555 unsigned cchStr = 0;
556
557 Assert(cueTokenizerGetCh(pTokenizer) == '\"');
558 cueTokenizerSkipCh(pTokenizer); /* Skip " */
559
560 pToken->enmType = CUETOKENTYPE_STRING;
561 pToken->Type.String.psz = pTokenizer->pszInput;
562
563 while (cueTokenizerGetCh(pTokenizer) != '\"')
564 {
565 cchStr++;
566 cueTokenizerSkipCh(pTokenizer);
567 }
568
569 cueTokenizerSkipCh(pTokenizer); /* Skip closing " */
570
571 pToken->Type.String.cch = cchStr;
572}
573
574/**
575 * Get the end of stream token.
576 *
577 * @returns nothing.
578 * @param pTokenizer The tokenizer state.
579 * @param pToken The uninitialized token.
580 */
581static void cueTokenizerGetEos(PCUETOKENIZER pTokenizer, PCUETOKEN pToken)
582{
583 Assert(cueTokenizerGetCh(pTokenizer) == '\0'); RT_NOREF(pTokenizer);
584
585 pToken->enmType = CUETOKENTYPE_EOS;
586}
587
588/**
589 * Read the next token from the tokenizer stream.
590 *
591 * @returns nothing.
592 * @param pTokenizer The tokenizer to read from.
593 * @param pToken Uninitialized token to fill the token data into.
594 */
595static void cueTokenizerReadNextToken(PCUETOKENIZER pTokenizer, PCUETOKEN pToken)
596{
597 /* Skip all eventually existing whitespace, newlines and comments first. */
598 cueTokenizerSkipWhitespace(pTokenizer);
599
600 char ch = cueTokenizerGetCh(pTokenizer);
601 if (RT_C_IS_ALPHA(ch))
602 cueTokenizerGetKeyword(pTokenizer, pToken);
603 else if (RT_C_IS_DIGIT(ch))
604 cueTokenizerGetIntegerOrMsf(pTokenizer, pToken);
605 else if (ch == '\"')
606 cueTokenizerGetStringConst(pTokenizer, pToken);
607 else if (ch == '\0')
608 cueTokenizerGetEos(pTokenizer, pToken);
609 else
610 pToken->enmType = CUETOKENTYPE_ERROR;
611}
612
613/**
614 * Create a new tokenizer.
615 *
616 * @returns Pointer to the new tokenizer state on success.
617 * NULL if out of memory.
618 * @param pszInput The input to create the tokenizer for.
619 */
620static PCUETOKENIZER cueTokenizerCreate(const char *pszInput)
621{
622 PCUETOKENIZER pTokenizer = (PCUETOKENIZER)RTMemAllocZ(sizeof(CUETOKENIZER));
623 if (pTokenizer)
624 {
625 pTokenizer->pszInput = pszInput;
626 pTokenizer->pTokenCurr = &pTokenizer->Token1;
627 pTokenizer->pTokenNext = &pTokenizer->Token2;
628 /* Fill the tokenizer with two first tokens. */
629 cueTokenizerReadNextToken(pTokenizer, pTokenizer->pTokenCurr);
630 cueTokenizerReadNextToken(pTokenizer, pTokenizer->pTokenNext);
631 }
632
633 return pTokenizer;
634}
635
636/**
637 * Get the current token in the input stream.
638 *
639 * @returns Pointer to the next token in the stream.
640 * @param pTokenizer The tokenizer to destroy.
641 */
642DECLINLINE(PCCUETOKEN) cueTokenizerGetToken(PCUETOKENIZER pTokenizer)
643{
644 return pTokenizer->pTokenCurr;
645}
646
647/**
648 * Get the class of the current token.
649 *
650 * @returns Class of the current token.
651 * @param pTokenizer The tokenizer state.
652 */
653DECLINLINE(CUETOKENTYPE) cueTokenizerGetTokenType(PCUETOKENIZER pTokenizer)
654{
655 return pTokenizer->pTokenCurr->enmType;
656}
657
658/**
659 * Consume the current token advancing to the next in the stream.
660 *
661 * @returns nothing.
662 * @param pTokenizer The tokenizer state.
663 */
664static void cueTokenizerConsume(PCUETOKENIZER pTokenizer)
665{
666 PCUETOKEN pTokenTmp = pTokenizer->pTokenCurr;
667
668 /* Switch next token to current token and read in the next token. */
669 pTokenizer->pTokenCurr = pTokenizer->pTokenNext;
670 pTokenizer->pTokenNext = pTokenTmp;
671 cueTokenizerReadNextToken(pTokenizer, pTokenizer->pTokenNext);
672}
673
674/**
675 * Check whether the next token in the input stream is a keyword and matches the given
676 * keyword.
677 *
678 * @returns true if the token matched.
679 * false otherwise.
680 * @param pTokenizer The tokenizer state.
681 * @param enmKeyword The keyword to check against.
682 */
683static bool cueTokenizerIsKeywordEqual(PCUETOKENIZER pTokenizer, CUEKEYWORD enmKeyword)
684{
685 PCCUETOKEN pToken = cueTokenizerGetToken(pTokenizer);
686
687 if ( pToken->enmType == CUETOKENTYPE_KEYWORD
688 && pToken->Type.Keyword.enmKeyword == enmKeyword)
689 return true;
690
691 return false;
692}
693
694/**
695 * Check whether the next token in the input stream is a keyword and matches the given
696 * keyword and skips it.
697 *
698 * @returns true if the token matched and was skipped.
699 * false otherwise.
700 * @param pTokenizer The tokenizer state.
701 * @param enmKeyword The keyword to check against.
702 */
703static bool cueTokenizerSkipIfIsKeywordEqual(PCUETOKENIZER pTokenizer, CUEKEYWORD enmKeyword)
704{
705 bool fEqual = cueTokenizerIsKeywordEqual(pTokenizer, enmKeyword);
706 if (fEqual)
707 cueTokenizerConsume(pTokenizer);
708
709 return fEqual;
710}
711
712/**
713 * Duplicates the string of the current token and consumes it.
714 *
715 * @returns VBox status code.
716 * @param pTokenizer The tokenizer state.
717 * @param ppszStr Where to store the pointer to the duplicated string on success.
718 * Free with RTStrFree().
719 */
720static int cueTokenizerConsumeStringDup(PCUETOKENIZER pTokenizer, char **ppszStr)
721{
722 int rc = VINF_SUCCESS;
723 Assert(cueTokenizerGetTokenType(pTokenizer) == CUETOKENTYPE_STRING);
724
725 *ppszStr = RTStrDupN(pTokenizer->pTokenCurr->Type.String.psz,
726 pTokenizer->pTokenCurr->Type.String.cch);
727 if (!*ppszStr)
728 rc = VERR_NO_STR_MEMORY;
729
730 cueTokenizerConsume(pTokenizer);
731 return rc;
732}
733
734/**
735 * Consumes an integer token returning the value.
736 *
737 * @returns Integer value in the token.
738 * @param pTokenizer The tokenizer state.
739 */
740static uint64_t cueTokenizerConsumeInteger(PCUETOKENIZER pTokenizer)
741{
742 uint64_t u64 = 0;
743 Assert(cueTokenizerGetTokenType(pTokenizer) == CUETOKENTYPE_INTEGER_UNSIGNED);
744
745 u64 = pTokenizer->pTokenCurr->Type.Int.u64;
746 cueTokenizerConsume(pTokenizer);
747 return u64;
748}
749
750/**
751 * Parses and skips the remaining string part of a directive.
752 *
753 * @returns VBox status code.
754 * @param pThis The CUE image state.
755 * @param pTokenizer The tokenizer state.
756 * @param pszDirective The directive we skip the string part for.
757 */
758static int cueParseAndSkipStringRemainder(PCUEIMAGE pThis, PCUETOKENIZER pTokenizer,
759 const char *pszDirective)
760{
761 int rc = VINF_SUCCESS;
762
763 if (cueTokenizerGetTokenType(pTokenizer) == CUETOKENTYPE_STRING)
764 cueTokenizerConsume(pTokenizer);
765 else
766 rc = vdIfError(pThis->pIfError, VERR_NOT_SUPPORTED, RT_SRC_POS,
767 N_("CUE: Error parsing '%s', expected string for %s directive"), pThis->pszFilename,
768 pszDirective);
769
770 return rc;
771}
772
773/**
774 * Parses and skips the remaining MSF part of a directive.
775 *
776 * @returns VBox status code.
777 * @param pThis The CUE image state.
778 * @param pTokenizer The tokenizer state.
779 * @param pszDirective The directive we skip the string part for.
780 */
781static int cueParseAndSkipMsfRemainder(PCUEIMAGE pThis, PCUETOKENIZER pTokenizer,
782 const char *pszDirective)
783{
784 int rc = VINF_SUCCESS;
785
786 if (cueTokenizerGetTokenType(pTokenizer) == CUETOKENTYPE_MSF)
787 cueTokenizerConsume(pTokenizer);
788 else
789 rc = vdIfError(pThis->pIfError, VERR_NOT_SUPPORTED, RT_SRC_POS,
790 N_("CUE: Error parsing '%s', expected MSF location for %s directive"), pThis->pszFilename,
791 pszDirective);
792
793 return rc;
794}
795
796/**
797 * Parses the remainder of a INDEX directive.
798 *
799 * @returns VBox status code.
800 * @param pThis The CUE image state.
801 * @param pTokenizer The tokenizer state.
802 * @param pu8Index Where to store the parsed index number on success.
803 * @param pu64Lba Where to store the parsed positional information on success.
804 */
805static int cueParseIndex(PCUEIMAGE pThis, PCUETOKENIZER pTokenizer,
806 uint8_t *pu8Index, uint64_t *pu64Lba)
807{
808 int rc = VINF_SUCCESS;
809
810 /*
811 * The index consists of the index number and positional information in MSF format.
812 */
813 if (cueTokenizerGetTokenType(pTokenizer) == CUETOKENTYPE_INTEGER_UNSIGNED)
814 {
815 uint64_t u64Index = cueTokenizerConsumeInteger(pTokenizer);
816 if (u64Index <= 99)
817 {
818 /* Parse the position. */
819 if (cueTokenizerGetTokenType(pTokenizer) == CUETOKENTYPE_MSF)
820 {
821 PCCUETOKEN pToken = cueTokenizerGetToken(pTokenizer);
822 uint8_t abMsf[3];
823 abMsf[0] = pToken->Type.Msf.u8Minute;
824 abMsf[1] = pToken->Type.Msf.u8Second;
825 abMsf[2] = pToken->Type.Msf.u8Frame;
826
827 *pu8Index = (uint8_t)u64Index;
828 *pu64Lba = scsiMSF2LBA(&abMsf[0]);
829 cueTokenizerConsume(pTokenizer);
830 }
831 else
832 rc = vdIfError(pThis->pIfError, VERR_NOT_SUPPORTED, RT_SRC_POS,
833 N_("CUE: Error parsing '%s', expected MSF location"), pThis->pszFilename);
834 }
835 else
836 rc = vdIfError(pThis->pIfError, VERR_NOT_SUPPORTED, RT_SRC_POS,
837 N_("CUE: Error parsing '%s', index number must be between 01 and 99"), pThis->pszFilename);
838 }
839 else
840 rc = vdIfError(pThis->pIfError, VERR_NOT_SUPPORTED, RT_SRC_POS,
841 N_("CUE: Error parsing '%s', expected index number after INDEX directive"), pThis->pszFilename);
842
843 return rc;
844}
845
846/**
847 * Parses the things coming below a TRACK directive.
848 *
849 * @returns VBox status code.
850 * @param pThis The CUE image state.
851 * @param pTokenizer The tokenizer state.
852 * @param pu64LbaStart Where to store the starting LBA for this track on success.
853 */
854static int cueParseTrackNesting(PCUEIMAGE pThis, PCUETOKENIZER pTokenizer, uint64_t *pu64LbaStart)
855{
856 int rc = VINF_SUCCESS;
857 bool fSeenInitialIndex = false;
858
859 do
860 {
861 if ( cueTokenizerIsKeywordEqual(pTokenizer, CUEKEYWORD_TRACK)
862 || cueTokenizerGetTokenType(pTokenizer) == CUETOKENTYPE_EOS)
863 break;
864
865 if (cueTokenizerGetTokenType(pTokenizer) == CUETOKENTYPE_KEYWORD)
866 {
867 if (cueTokenizerSkipIfIsKeywordEqual(pTokenizer, CUEKEYWORD_TITLE))
868 rc = cueParseAndSkipStringRemainder(pThis, pTokenizer, "TITLE");
869 else if (cueTokenizerSkipIfIsKeywordEqual(pTokenizer, CUEKEYWORD_PERFORMER))
870 rc = cueParseAndSkipStringRemainder(pThis, pTokenizer, "PERFORMER");
871 else if (cueTokenizerSkipIfIsKeywordEqual(pTokenizer, CUEKEYWORD_PREGAP))
872 rc = cueParseAndSkipMsfRemainder(pThis, pTokenizer, "PREGAP");
873 else if (cueTokenizerSkipIfIsKeywordEqual(pTokenizer, CUEKEYWORD_POSTGAP))
874 rc = cueParseAndSkipMsfRemainder(pThis, pTokenizer, "POSTGAP");
875 else if (cueTokenizerSkipIfIsKeywordEqual(pTokenizer, CUEKEYWORD_INDEX))
876 {
877 uint8_t u8Index = 0;
878 uint64_t u64Lba = 0;
879 rc = cueParseIndex(pThis, pTokenizer, &u8Index, &u64Lba);
880 if ( RT_SUCCESS(rc)
881 && u8Index == 1)
882 {
883 if (!fSeenInitialIndex)
884 {
885 fSeenInitialIndex = true;
886 *pu64LbaStart = u64Lba;
887 }
888 else
889 rc = vdIfError(pThis->pIfError, VERR_NOT_SUPPORTED, RT_SRC_POS,
890 N_("CUE: Error parsing '%s', multiple INDEX 01 directives"), pThis->pszFilename);
891 }
892 }
893 else
894 rc = vdIfError(pThis->pIfError, VERR_NOT_SUPPORTED, RT_SRC_POS,
895 N_("CUE: Error parsing '%s', unexpected directive for TRACK found"), pThis->pszFilename);
896 }
897 else
898 rc = vdIfError(pThis->pIfError, VERR_NOT_SUPPORTED, RT_SRC_POS,
899 N_("CUE: Error parsing '%s', expected a CUE sheet keyword"), pThis->pszFilename);
900 }
901 while (RT_SUCCESS(rc));
902
903 if ( RT_SUCCESS(rc)
904 && !fSeenInitialIndex)
905 rc = vdIfError(pThis->pIfError, VERR_NOT_SUPPORTED, RT_SRC_POS,
906 N_("CUE: Error parsing '%s', no initial INDEX directive for this track"), pThis->pszFilename);
907
908 return rc;
909}
910
911/**
912 * Parses the remainder of a TRACK directive.
913 *
914 * @returns VBox status code.
915 * @param pThis The CUE image state.
916 * @param pTokenizer The tokenizer state.
917 */
918static int cueParseTrack(PCUEIMAGE pThis, PCUETOKENIZER pTokenizer)
919{
920 int rc = VINF_SUCCESS;
921
922 /*
923 * A track consists of the track number and data type followed by a list of indexes
924 * and other metadata like title and performer we don't care about.
925 */
926 if (cueTokenizerGetTokenType(pTokenizer) == CUETOKENTYPE_INTEGER_UNSIGNED)
927 {
928 uint64_t u64Track = cueTokenizerConsumeInteger(pTokenizer);
929 if (u64Track <= 99)
930 {
931 /* Parse the data mode. */
932 if (cueTokenizerGetTokenType(pTokenizer) == CUETOKENTYPE_KEYWORD)
933 {
934 CUEKEYWORD enmDataMode = pTokenizer->pTokenCurr->Type.Keyword.enmKeyword;
935 if ( cueTokenizerSkipIfIsKeywordEqual(pTokenizer, CUEKEYWORD_AUDIO)
936 || cueTokenizerSkipIfIsKeywordEqual(pTokenizer, CUEKEYWORD_MODE1_2048)
937 || cueTokenizerSkipIfIsKeywordEqual(pTokenizer, CUEKEYWORD_MODE1_2352))
938 {
939 /*
940 * Parse everything coming below the track (index points, etc.), we only need to find
941 * the starting point.
942 */
943 uint64_t uLbaStart = 0;
944 rc = cueParseTrackNesting(pThis, pTokenizer, &uLbaStart);
945 if (RT_SUCCESS(rc))
946 {
947 /* Create a new region for this track. */
948 RT_NOREF1(enmDataMode);
949 rc = cueEnsureRegionListSize(pThis, u64Track);
950 if (RT_SUCCESS(rc))
951 {
952 PVDREGIONDESC pRegion = &pThis->pRegionList->aRegions[u64Track - 1];
953 pRegion->offRegion = uLbaStart;
954 if (enmDataMode == CUEKEYWORD_MODE1_2352)
955 {
956 pRegion->cbBlock = 2352;
957 pRegion->enmDataForm = VDREGIONDATAFORM_MODE1_2352;
958 }
959 else if (enmDataMode == CUEKEYWORD_AUDIO)
960 {
961 pRegion->cbBlock = 2352;
962 pRegion->enmDataForm = VDREGIONDATAFORM_CDDA;
963 }
964 else
965 {
966 pRegion->cbBlock = 2048;
967 pRegion->enmDataForm = VDREGIONDATAFORM_MODE1_2048;
968 }
969 pRegion->enmMetadataForm = VDREGIONMETADATAFORM_NONE;
970 pRegion->cbData = pRegion->cbBlock;
971 pRegion->cbMetadata = 0;
972 }
973 else
974 rc = vdIfError(pThis->pIfError, rc, RT_SRC_POS,
975 N_("CUE: Failed to allocate memory for the track list for '%s'"),
976 pThis->pszFilename);
977 }
978 }
979 else
980 rc = vdIfError(pThis->pIfError, VERR_NOT_SUPPORTED, RT_SRC_POS,
981 N_("CUE: Error parsing '%s', the data mode is not supported"), pThis->pszFilename);
982 }
983 else
984 rc = vdIfError(pThis->pIfError, VERR_NOT_SUPPORTED, RT_SRC_POS,
985 N_("CUE: Error parsing '%s', expected data mode"), pThis->pszFilename);
986 }
987 else
988 rc = vdIfError(pThis->pIfError, VERR_NOT_SUPPORTED, RT_SRC_POS,
989 N_("CUE: Error parsing '%s', track number must be between 01 and 99"), pThis->pszFilename);
990 }
991 else
992 rc = vdIfError(pThis->pIfError, VERR_NOT_SUPPORTED, RT_SRC_POS,
993 N_("CUE: Error parsing '%s', expected track number after TRACK directive"), pThis->pszFilename);
994
995 return rc;
996}
997
998/**
999 * Parses a list of tracks which must come after a FILE directive.
1000 *
1001 * @returns VBox status code.
1002 * @param pThis The CUE image state.
1003 * @param pTokenizer The tokenizer state.
1004 */
1005static int cueParseTrackList(PCUEIMAGE pThis, PCUETOKENIZER pTokenizer)
1006{
1007 int rc = VINF_SUCCESS;
1008
1009 while ( RT_SUCCESS(rc)
1010 && cueTokenizerSkipIfIsKeywordEqual(pTokenizer, CUEKEYWORD_TRACK))
1011 rc = cueParseTrack(pThis, pTokenizer);
1012
1013 return rc;
1014}
1015
1016/**
1017 * Parses the remainder of a FILE directive.
1018 *
1019 * @returns VBox status code.
1020 * @param pThis The CUE image state.
1021 * @param pTokenizer The tokenizer state.
1022 */
1023static int cueParseFile(PCUEIMAGE pThis, PCUETOKENIZER pTokenizer)
1024{
1025 int rc = VINF_SUCCESS;
1026
1027 /* First must come a string constant followed by a keyword giving the file type. */
1028 if (cueTokenizerGetTokenType(pTokenizer) == CUETOKENTYPE_STRING)
1029 {
1030 rc = cueTokenizerConsumeStringDup(pTokenizer, &pThis->pszDataFilename);
1031 if (RT_SUCCESS(rc))
1032 {
1033 if (cueTokenizerGetTokenType(pTokenizer) == CUETOKENTYPE_KEYWORD)
1034 {
1035 if (cueTokenizerSkipIfIsKeywordEqual(pTokenizer, CUEKEYWORD_BINARY))
1036 {
1037 pThis->fLittleEndian = true;
1038 rc = cueParseTrackList(pThis, pTokenizer);
1039 }
1040 else if (cueTokenizerSkipIfIsKeywordEqual(pTokenizer, CUEKEYWORD_MOTOROLA))
1041 {
1042 pThis->fLittleEndian = false;
1043 rc = cueParseTrackList(pThis, pTokenizer);
1044 }
1045 else
1046 rc = vdIfError(pThis->pIfError, VERR_NOT_SUPPORTED, RT_SRC_POS,
1047 N_("CUE: Error parsing '%s', the file type is not supported (only BINARY)"), pThis->pszFilename);
1048 }
1049 else
1050 rc = vdIfError(pThis->pIfError, rc, RT_SRC_POS,
1051 N_("CUE: Error parsing '%s', expected file type"), pThis->pszFilename);
1052 }
1053 else
1054 rc = vdIfError(pThis->pIfError, rc, RT_SRC_POS,
1055 N_("CUE: Error parsing '%s', failed to allocate memory for filename"), pThis->pszFilename);
1056 }
1057 else
1058 rc = vdIfError(pThis->pIfError, VERR_NOT_SUPPORTED, RT_SRC_POS,
1059 N_("CUE: Error parsing '%s', expected filename after FILE directive"), pThis->pszFilename);
1060
1061 return rc;
1062}
1063
1064/**
1065 * Parses the keyword in the given tokenizer.
1066 *
1067 * @returns VBox status code.
1068 * @param pThis The CUE image state.
1069 * @param pTokenizer The tokenizer state.
1070 */
1071static int cueParseKeyword(PCUEIMAGE pThis, PCUETOKENIZER pTokenizer)
1072{
1073 int rc = VINF_SUCCESS;
1074
1075 if (cueTokenizerSkipIfIsKeywordEqual(pTokenizer, CUEKEYWORD_FILE))
1076 rc = cueParseFile(pThis, pTokenizer);
1077 else if (cueTokenizerSkipIfIsKeywordEqual(pTokenizer, CUEKEYWORD_TITLE))
1078 rc = cueParseAndSkipStringRemainder(pThis, pTokenizer, "TITLE");
1079 else if (cueTokenizerSkipIfIsKeywordEqual(pTokenizer, CUEKEYWORD_PERFORMER))
1080 rc = cueParseAndSkipStringRemainder(pThis, pTokenizer, "PERFORMER");
1081 else if (cueTokenizerSkipIfIsKeywordEqual(pTokenizer, CUEKEYWORD_SONGWRITER))
1082 rc = cueParseAndSkipStringRemainder(pThis, pTokenizer, "SONGWRITER");
1083 else /* Skip all other keywords we don't need/support. */
1084 cueTokenizerConsume(pTokenizer);
1085
1086 return rc;
1087}
1088
1089
1090/**
1091 * Parses the CUE sheet from the given tokenizer.
1092 *
1093 * @returns VBox status code.
1094 * @param pThis The CUE image state.
1095 * @param pTokenizer The tokenizer state.
1096 */
1097static int cueParseFromTokenizer(PCUEIMAGE pThis, PCUETOKENIZER pTokenizer)
1098{
1099 int rc = VINF_SUCCESS;
1100
1101 LogFlowFunc(("pThis=%p\n", pThis));
1102
1103 /* We don't support multiple FILE directives for now. */
1104 if (cueTokenizerGetTokenType(pTokenizer) == CUETOKENTYPE_KEYWORD)
1105 rc = cueParseKeyword(pThis, pTokenizer);
1106 else
1107 rc = vdIfError(pThis->pIfError, VERR_NOT_SUPPORTED, RT_SRC_POS,
1108 N_("CUE: Error parsing '%s', expected a keyword"), pThis->pszFilename);
1109
1110 if ( RT_SUCCESS(rc)
1111 && cueTokenizerGetTokenType(pTokenizer) != CUETOKENTYPE_EOS)
1112 rc = vdIfError(pThis->pIfError, VERR_NOT_SUPPORTED, RT_SRC_POS,
1113 N_("CUE: Error parsing '%s', expected end of stream"), pThis->pszFilename);
1114
1115 LogFlowFunc(("returns rc=%Rrc\n", rc));
1116 return rc;
1117}
1118
1119/**
1120 * Finalizes the track list of the image.
1121 *
1122 * @returns VBox status code.
1123 * @param pThis The CUE image state.
1124 * @param cbImage Size of the image data in bytes.
1125 */
1126static int cueTrackListFinalize(PCUEIMAGE pThis, uint64_t cbImage)
1127{
1128 int rc = VINF_SUCCESS;
1129
1130 if ( pThis->cTracksMax == 0
1131 || pThis->pRegionList->aRegions[0].offRegion == UINT64_MAX)
1132 return vdIfError(pThis->pIfError, VERR_NOT_SUPPORTED, RT_SRC_POS,
1133 N_("CUE: Error parsing '%s', detected empty track list"), pThis->pszFilename);
1134
1135 /*
1136 * Fixup the track list to contain the proper sizes now that we parsed all tracks,
1137 * check also that there are no gaps in the list.
1138 */
1139 uint32_t cTracks = 1;
1140 uint64_t offDisk = 0;
1141 for (uint32_t i = 1; i < pThis->cTracksMax; i++)
1142 {
1143 PVDREGIONDESC pRegion = &pThis->pRegionList->aRegions[i];
1144 PVDREGIONDESC pRegionPrev = &pThis->pRegionList->aRegions[i - 1];
1145 if (pRegion->offRegion != UINT64_MAX)
1146 {
1147 cTracks++;
1148 uint64_t cBlocks = pRegion->offRegion - (pRegionPrev->offRegion / pRegionPrev->cbBlock);
1149 pRegionPrev->cRegionBlocksOrBytes = pRegionPrev->cbBlock * cBlocks;
1150 offDisk += pRegionPrev->cRegionBlocksOrBytes;
1151
1152 if (cbImage < pRegionPrev->cRegionBlocksOrBytes)
1153 return vdIfError(pThis->pIfError, VERR_NOT_SUPPORTED, RT_SRC_POS,
1154 N_("CUE: Error parsing '%s', image file is too small for track list"),
1155 pThis->pszFilename);
1156
1157 cbImage -= pRegionPrev->cRegionBlocksOrBytes;
1158 pRegion->offRegion = offDisk;
1159 }
1160 else
1161 break;
1162 }
1163
1164 /* Fixup last track. */
1165 PVDREGIONDESC pRegion = &pThis->pRegionList->aRegions[cTracks - 1];
1166 pRegion->cRegionBlocksOrBytes = cbImage;
1167
1168 pThis->pRegionList->cRegions = cTracks;
1169 pThis->pRegionList->fFlags = 0;
1170
1171 /* Check that there are no gaps in the track list. */
1172 for (uint32_t i = cTracks; cTracks < pThis->cTracksMax; i++)
1173 {
1174 pRegion = &pThis->pRegionList->aRegions[i];
1175 if (pRegion->offRegion != UINT64_MAX)
1176 {
1177 rc = vdIfError(pThis->pIfError, VERR_NOT_SUPPORTED, RT_SRC_POS,
1178 N_("CUE: Error parsing '%s', detected gaps in the track list"), pThis->pszFilename);
1179 break;
1180 }
1181 }
1182
1183 return rc;
1184}
1185
1186/**
1187 * Internal. Free all allocated space for representing an image except pThis,
1188 * and optionally delete the image from disk.
1189 */
1190static int cueFreeImage(PCUEIMAGE pThis, bool fDelete)
1191{
1192 int rc = VINF_SUCCESS;
1193
1194 /* Freeing a never allocated image (e.g. because the open failed) is
1195 * not signalled as an error. After all nothing bad happens. */
1196 if (pThis)
1197 {
1198 if (pThis->pStorage)
1199 {
1200 rc = vdIfIoIntFileClose(pThis->pIfIo, pThis->pStorage);
1201 pThis->pStorage = NULL;
1202 }
1203
1204 if (pThis->pStorageData)
1205 {
1206 rc = vdIfIoIntFileClose(pThis->pIfIo, pThis->pStorageData);
1207 pThis->pStorageData = NULL;
1208 }
1209
1210 if (pThis->pRegionList)
1211 {
1212 RTMemFree(pThis->pRegionList);
1213 pThis->pRegionList = NULL;
1214 }
1215
1216 if (pThis->pszDataFilename)
1217 {
1218 RTStrFree(pThis->pszDataFilename);
1219 pThis->pszDataFilename = NULL;
1220 }
1221
1222 if (fDelete && pThis->pszFilename)
1223 vdIfIoIntFileDelete(pThis->pIfIo, pThis->pszFilename);
1224 }
1225
1226 LogFlowFunc(("returns %Rrc\n", rc));
1227 return rc;
1228}
1229
1230/**
1231 * Internal: Open an image, constructing all necessary data structures.
1232 */
1233static int cueOpenImage(PCUEIMAGE pThis, unsigned uOpenFlags)
1234{
1235 pThis->uOpenFlags = uOpenFlags;
1236
1237 pThis->pIfError = VDIfErrorGet(pThis->pVDIfsDisk);
1238 pThis->pIfIo = VDIfIoIntGet(pThis->pVDIfsImage);
1239 AssertPtrReturn(pThis->pIfIo, VERR_INVALID_PARAMETER);
1240
1241 /* Open the image. */
1242 int rc = vdIfIoIntFileOpen(pThis->pIfIo, pThis->pszFilename,
1243 VDOpenFlagsToFileOpenFlags(uOpenFlags,
1244 false /* fCreate */),
1245 &pThis->pStorage);
1246 if (RT_SUCCESS(rc))
1247 {
1248 uint64_t cbFile;
1249 /* The descriptor file shouldn't be huge, so limit ourselfs to 16KB for now. */
1250 rc = vdIfIoIntFileGetSize(pThis->pIfIo, pThis->pStorage, &cbFile);
1251 if ( RT_SUCCESS(rc)
1252 && cbFile <= _16K - 1)
1253 {
1254 char szInput[_16K];
1255 RT_ZERO(szInput);
1256
1257 rc = vdIfIoIntFileReadSync(pThis->pIfIo, pThis->pStorage, 0,
1258 &szInput, cbFile);
1259 if (RT_SUCCESS(rc))
1260 {
1261 RTStrPurgeEncoding(&szInput[0]);
1262 PCUETOKENIZER pTokenizer = cueTokenizerCreate(&szInput[0]);
1263 if (pTokenizer)
1264 {
1265 rc = cueParseFromTokenizer(pThis, pTokenizer);
1266 RTMemFree(pTokenizer);
1267 if (RT_SUCCESS(rc))
1268 {
1269 /* Open the backing file. */
1270 char szBackingFile[RTPATH_MAX];
1271 rc = RTStrCopy(&szBackingFile[0], sizeof(szBackingFile), pThis->pszFilename);
1272 if (RT_SUCCESS(rc))
1273 {
1274 RTPathStripFilename(&szBackingFile[0]);
1275 rc = RTPathAppend(&szBackingFile[0], sizeof(szBackingFile), pThis->pszDataFilename);
1276 if (RT_SUCCESS(rc))
1277 {
1278 rc = vdIfIoIntFileOpen(pThis->pIfIo, szBackingFile,
1279 VDOpenFlagsToFileOpenFlags(uOpenFlags,
1280 false /* fCreate */),
1281 &pThis->pStorageData);
1282 if (RT_SUCCESS(rc))
1283 {
1284 rc = vdIfIoIntFileGetSize(pThis->pIfIo, pThis->pStorageData, &cbFile);
1285 if (RT_SUCCESS(rc))
1286 rc = cueTrackListFinalize(pThis, cbFile);
1287 else
1288 rc = vdIfError(pThis->pIfError, rc, RT_SRC_POS,
1289 N_("CUE: Unable to query size of backing file '%s'"),
1290 szBackingFile);
1291 }
1292 else
1293 rc = vdIfError(pThis->pIfError, rc, RT_SRC_POS,
1294 N_("CUE: Unable to open backing file '%s'"),
1295 szBackingFile);
1296 }
1297 else
1298 rc = vdIfError(pThis->pIfError, rc, RT_SRC_POS,
1299 N_("CUE: Error constructing backing filename from '%s'"),
1300 pThis->pszFilename);
1301 }
1302 else
1303 rc = vdIfError(pThis->pIfError, rc, RT_SRC_POS,
1304 N_("CUE: Error constructing backing filename from '%s'"),
1305 pThis->pszFilename);
1306 }
1307 }
1308 }
1309 else
1310 rc = vdIfError(pThis->pIfError, rc, RT_SRC_POS, N_("CUE: Error reading '%s'"), pThis->pszFilename);
1311 }
1312 else if (RT_SUCCESS(rc))
1313 rc = vdIfError(pThis->pIfError, rc, RT_SRC_POS, N_("CUE: The descriptor file '%s' is too huge (%llu vs %llu)"),
1314 pThis->pszFilename, cbFile, _16K - 1);
1315 }
1316 /* else: Do NOT signal an appropriate error here, as the VD layer has the
1317 * choice of retrying the open if it failed. */
1318
1319 if (RT_FAILURE(rc))
1320 cueFreeImage(pThis, false);
1321 return rc;
1322}
1323
1324/**
1325 * Converts the data form enumeration to a string.
1326 *
1327 * @returns String name of the given data form.
1328 * @param enmDataForm The data form.
1329 */
1330static const char *cueRegionDataFormStringify(VDREGIONDATAFORM enmDataForm)
1331{
1332 switch (enmDataForm)
1333 {
1334 #define DATAFORM2STR(tag) case VDREGIONDATAFORM_##tag: return #tag
1335
1336 DATAFORM2STR(INVALID);
1337 DATAFORM2STR(RAW);
1338 DATAFORM2STR(CDDA);
1339 DATAFORM2STR(CDDA_PAUSE);
1340 DATAFORM2STR(MODE1_2048);
1341 DATAFORM2STR(MODE1_2352);
1342 DATAFORM2STR(MODE1_0);
1343 DATAFORM2STR(XA_2336);
1344 DATAFORM2STR(XA_2352);
1345 DATAFORM2STR(XA_0);
1346 DATAFORM2STR(MODE2_2336);
1347 DATAFORM2STR(MODE2_2352);
1348 DATAFORM2STR(MODE2_0);
1349
1350 #undef DATAFORM2STR
1351
1352 default:
1353 {
1354 AssertMsgFailed(("Unknown data form %d! forgot to add it to the switch?\n", enmDataForm));
1355 return "UNKNOWN!";
1356 }
1357 }
1358}
1359
1360/**
1361 * Converts the data form enumeration to a string.
1362 *
1363 * @returns String name of the given data form.
1364 * @param enmMetadataForm The metadata form.
1365 */
1366static const char *cueRegionMetadataFormStringify(VDREGIONMETADATAFORM enmMetadataForm)
1367{
1368 switch (enmMetadataForm)
1369 {
1370 #define METADATAFORM2STR(tag) case VDREGIONMETADATAFORM_##tag: return #tag
1371
1372 METADATAFORM2STR(INVALID);
1373 METADATAFORM2STR(RAW);
1374 METADATAFORM2STR(NONE);
1375
1376 #undef METADATAFORM2STR
1377
1378 default:
1379 {
1380 AssertMsgFailed(("Unknown metadata form %d! forgot to add it to the switch?\n", enmMetadataForm));
1381 return "UNKNOWN!";
1382 }
1383 }
1384}
1385
1386/**
1387 * Returns the region containing the given offset.
1388 *
1389 * @returns Pointer to the region or NULL if not found.
1390 * @param pThis The CUE image state.
1391 * @param uOffset The offset to look for.
1392 */
1393static PCVDREGIONDESC cueRegionQueryByOffset(PCUEIMAGE pThis, uint64_t uOffset)
1394{
1395 for (uint32_t i = 0; i < pThis->pRegionList->cRegions; i++)
1396 {
1397 PCVDREGIONDESC pRegion = &pThis->pRegionList->aRegions[i];
1398 if ( pRegion->offRegion <= uOffset
1399 && pRegion->offRegion + pRegion->cRegionBlocksOrBytes > uOffset)
1400 return pRegion;
1401 }
1402
1403 return NULL;
1404}
1405
1406/** @copydoc VDIMAGEBACKEND::pfnProbe */
1407static DECLCALLBACK(int) cueProbe(const char *pszFilename, PVDINTERFACE pVDIfsDisk,
1408 PVDINTERFACE pVDIfsImage, VDTYPE *penmType)
1409{
1410 RT_NOREF1(pVDIfsDisk);
1411 LogFlowFunc(("pszFilename=\"%s\" pVDIfsDisk=%#p pVDIfsImage=%#p\n", pszFilename, pVDIfsDisk, pVDIfsImage));
1412 int rc = VINF_SUCCESS;
1413
1414 AssertReturn((VALID_PTR(pszFilename) && *pszFilename), VERR_INVALID_PARAMETER);
1415
1416 PCUEIMAGE pThis = (PCUEIMAGE)RTMemAllocZ(sizeof(CUEIMAGE));
1417 if (RT_LIKELY(pThis))
1418 {
1419 pThis->pszFilename = pszFilename;
1420 pThis->pStorage = NULL;
1421 pThis->pVDIfsDisk = pVDIfsDisk;
1422 pThis->pVDIfsImage = pVDIfsImage;
1423
1424 rc = cueOpenImage(pThis, VD_OPEN_FLAGS_INFO | VD_OPEN_FLAGS_READONLY);
1425 cueFreeImage(pThis, false);
1426 RTMemFree(pThis);
1427
1428 if (RT_SUCCESS(rc))
1429 *penmType = VDTYPE_OPTICAL_DISC;
1430 else
1431 rc = VERR_VD_GEN_INVALID_HEADER;
1432 }
1433 else
1434 rc = VERR_NO_MEMORY;
1435
1436 LogFlowFunc(("returns %Rrc\n", rc));
1437 return rc;
1438}
1439
1440/** @copydoc VDIMAGEBACKEND::pfnOpen */
1441static DECLCALLBACK(int) cueOpen(const char *pszFilename, unsigned uOpenFlags,
1442 PVDINTERFACE pVDIfsDisk, PVDINTERFACE pVDIfsImage,
1443 VDTYPE enmType, void **ppBackendData)
1444{
1445 LogFlowFunc(("pszFilename=\"%s\" uOpenFlags=%#x pVDIfsDisk=%#p pVDIfsImage=%#p enmType=%u ppBackendData=%#p\n",
1446 pszFilename, uOpenFlags, pVDIfsDisk, pVDIfsImage, enmType, ppBackendData));
1447 int rc;
1448 PCUEIMAGE pThis;
1449
1450 /* Check open flags. All valid flags are supported. */
1451 AssertReturn(!(uOpenFlags & ~VD_OPEN_FLAGS_MASK), VERR_INVALID_PARAMETER);
1452 AssertReturn((VALID_PTR(pszFilename) && *pszFilename), VERR_INVALID_PARAMETER);
1453 AssertReturn(enmType == VDTYPE_OPTICAL_DISC, VERR_NOT_SUPPORTED);
1454
1455 pThis = (PCUEIMAGE)RTMemAllocZ(sizeof(CUEIMAGE));
1456 if (RT_LIKELY(pThis))
1457 {
1458 pThis->pszFilename = pszFilename;
1459 pThis->pStorage = NULL;
1460 pThis->pVDIfsDisk = pVDIfsDisk;
1461 pThis->pVDIfsImage = pVDIfsImage;
1462
1463 rc = cueOpenImage(pThis, uOpenFlags);
1464 if (RT_SUCCESS(rc))
1465 *ppBackendData = pThis;
1466 else
1467 RTMemFree(pThis);
1468 }
1469 else
1470 rc = VERR_NO_MEMORY;
1471
1472 LogFlowFunc(("returns %Rrc (pBackendData=%#p)\n", rc, *ppBackendData));
1473 return rc;
1474}
1475
1476/** @copydoc VDIMAGEBACKEND::pfnClose */
1477static DECLCALLBACK(int) cueClose(void *pBackendData, bool fDelete)
1478{
1479 LogFlowFunc(("pBackendData=%#p fDelete=%d\n", pBackendData, fDelete));
1480 PCUEIMAGE pThis = (PCUEIMAGE)pBackendData;
1481 int rc = cueFreeImage(pThis, fDelete);
1482 RTMemFree(pThis);
1483
1484 LogFlowFunc(("returns %Rrc\n", rc));
1485 return rc;
1486}
1487
1488/** @copydoc VDIMAGEBACKEND::pfnRead */
1489static DECLCALLBACK(int) cueRead(void *pBackendData, uint64_t uOffset, size_t cbToRead,
1490 PVDIOCTX pIoCtx, size_t *pcbActuallyRead)
1491{
1492 LogFlowFunc(("pBackendData=%#p uOffset=%llu cbToRead=%zu pIoCtx=%#p pcbActuallyRead=%#p\n",
1493 pBackendData, uOffset, cbToRead, pIoCtx, pcbActuallyRead));
1494 int rc = VINF_SUCCESS;
1495 PCUEIMAGE pThis = (PCUEIMAGE)pBackendData;
1496
1497 /* Get the region */
1498 PCVDREGIONDESC pRegion = cueRegionQueryByOffset(pThis, uOffset);
1499 if (pRegion)
1500 {
1501 /* Clip read size to remain in the region (not necessary I think). */
1502 uint64_t offRead = uOffset - pRegion->offRegion;
1503
1504 cbToRead = RT_MIN(cbToRead, pRegion->cRegionBlocksOrBytes - offRead);
1505 Assert(!(cbToRead % pRegion->cbBlock));
1506
1507 /* Need to convert audio data samples to big endian. */
1508 if ( pRegion->enmDataForm == VDREGIONDATAFORM_CDDA
1509 && pThis->fLittleEndian)
1510 {
1511 *pcbActuallyRead = cbToRead;
1512
1513 while (cbToRead)
1514 {
1515 RTSGSEG Segment;
1516 unsigned cSegments = 1;
1517 size_t cbSeg = 0;
1518
1519 cbSeg = vdIfIoIntIoCtxSegArrayCreate(pThis->pIfIo, pIoCtx, &Segment,
1520 &cSegments, cbToRead);
1521
1522 rc = vdIfIoIntFileReadSync(pThis->pIfIo, pThis->pStorageData, uOffset, Segment.pvSeg, cbSeg);
1523 if (RT_FAILURE(rc))
1524 break;
1525
1526 uint16_t *pu16Buf = (uint16_t *)Segment.pvSeg;
1527 for (uint32_t i = 0; i < cbSeg / sizeof(uint16_t); i++)
1528 {
1529 *pu16Buf = RT_BSWAP_U16(*pu16Buf);
1530 pu16Buf++;
1531 }
1532
1533 cbToRead -= RT_MIN(cbToRead, cbSeg);
1534 uOffset += cbSeg;
1535 }
1536 }
1537 else
1538 {
1539 rc = vdIfIoIntFileReadUser(pThis->pIfIo, pThis->pStorageData, uOffset,
1540 pIoCtx, cbToRead);
1541 if (RT_SUCCESS(rc))
1542 *pcbActuallyRead = cbToRead;
1543 }
1544 }
1545 else
1546 rc = VERR_INVALID_PARAMETER;
1547
1548 return rc;
1549}
1550
1551/** @copydoc VDIMAGEBACKEND::pfnWrite */
1552static DECLCALLBACK(int) cueWrite(void *pBackendData, uint64_t uOffset, size_t cbToWrite,
1553 PVDIOCTX pIoCtx, size_t *pcbWriteProcess, size_t *pcbPreRead,
1554 size_t *pcbPostRead, unsigned fWrite)
1555{
1556 RT_NOREF7(uOffset, cbToWrite, pIoCtx, pcbWriteProcess, pcbPreRead, pcbPostRead, fWrite);
1557 LogFlowFunc(("pBackendData=%#p uOffset=%llu pIoCtx=%#p cbToWrite=%zu pcbWriteProcess=%#p pcbPreRead=%#p pcbPostRead=%#p\n",
1558 pBackendData, uOffset, pIoCtx, cbToWrite, pcbWriteProcess, pcbPreRead, pcbPostRead));
1559 PCUEIMAGE pThis = (PCUEIMAGE)pBackendData;
1560 int rc;
1561
1562 AssertPtr(pThis);
1563
1564 if (pThis->uOpenFlags & VD_OPEN_FLAGS_READONLY)
1565 rc = VERR_VD_IMAGE_READ_ONLY;
1566 else
1567 rc = VERR_NOT_SUPPORTED;
1568
1569 LogFlowFunc(("returns %Rrc\n", rc));
1570 return rc;
1571}
1572
1573/** @copydoc VDIMAGEBACKEND::pfnFlush */
1574static DECLCALLBACK(int) cueFlush(void *pBackendData, PVDIOCTX pIoCtx)
1575{
1576 RT_NOREF2(pBackendData, pIoCtx);
1577
1578 return VINF_SUCCESS;
1579}
1580
1581/** @copydoc VDIMAGEBACKEND::pfnGetVersion */
1582static DECLCALLBACK(unsigned) cueGetVersion(void *pBackendData)
1583{
1584 LogFlowFunc(("pBackendData=%#p\n", pBackendData));
1585 PCUEIMAGE pThis = (PCUEIMAGE)pBackendData;
1586
1587 AssertPtrReturn(pThis, 0);
1588
1589 return 1;
1590}
1591
1592/** @copydoc VDIMAGEBACKEND::pfnGetSectorSize */
1593static DECLCALLBACK(uint32_t) cueGetSectorSize(void *pBackendData)
1594{
1595 LogFlowFunc(("pBackendData=%#p\n", pBackendData));
1596 PCUEIMAGE pThis = (PCUEIMAGE)pBackendData;
1597 uint32_t cb = 0;
1598
1599 AssertPtrReturn(pThis, 0);
1600
1601 LogFlowFunc(("returns %u\n", cb));
1602 return cb;
1603}
1604
1605/** @copydoc VDIMAGEBACKEND::pfnGetSize */
1606static DECLCALLBACK(uint64_t) cueGetSize(void *pBackendData)
1607{
1608 LogFlowFunc(("pBackendData=%#p\n", pBackendData));
1609 PCUEIMAGE pThis = (PCUEIMAGE)pBackendData;
1610
1611 AssertPtrReturn(pThis, 0);
1612
1613 PCVDREGIONDESC pRegion = &pThis->pRegionList->aRegions[pThis->pRegionList->cRegions - 1];
1614 uint64_t cb = pRegion->offRegion + pRegion->cRegionBlocksOrBytes;
1615
1616 LogFlowFunc(("returns %llu\n", cb));
1617 return cb;
1618}
1619
1620/** @copydoc VDIMAGEBACKEND::pfnGetFileSize */
1621static DECLCALLBACK(uint64_t) cueGetFileSize(void *pBackendData)
1622{
1623 LogFlowFunc(("pBackendData=%#p\n", pBackendData));
1624 PCUEIMAGE pThis = (PCUEIMAGE)pBackendData;
1625
1626 AssertPtrReturn(pThis, 0);
1627
1628 uint64_t cbFile = 0;
1629 if (pThis->pStorage)
1630 {
1631 int rc = vdIfIoIntFileGetSize(pThis->pIfIo, pThis->pStorageData, &cbFile);
1632 if (RT_FAILURE(rc))
1633 cbFile = 0; /* Make sure it is 0 */
1634 }
1635
1636 LogFlowFunc(("returns %lld\n", cbFile));
1637 return cbFile;
1638}
1639
1640/** @copydoc VDIMAGEBACKEND::pfnGetPCHSGeometry */
1641static DECLCALLBACK(int) cueGetPCHSGeometry(void *pBackendData,
1642 PVDGEOMETRY pPCHSGeometry)
1643{
1644 RT_NOREF1(pPCHSGeometry);
1645 LogFlowFunc(("pBackendData=%#p pPCHSGeometry=%#p\n", pBackendData, pPCHSGeometry));
1646 PCUEIMAGE pThis = (PCUEIMAGE)pBackendData;
1647 int rc = VINF_SUCCESS;
1648
1649 AssertPtrReturn(pThis, VERR_VD_NOT_OPENED);
1650
1651 rc = VERR_NOT_SUPPORTED;
1652
1653 LogFlowFunc(("returns %Rrc (PCHS=%u/%u/%u)\n", rc, pPCHSGeometry->cCylinders, pPCHSGeometry->cHeads, pPCHSGeometry->cSectors));
1654 return rc;
1655}
1656
1657/** @copydoc VDIMAGEBACKEND::pfnSetPCHSGeometry */
1658static DECLCALLBACK(int) cueSetPCHSGeometry(void *pBackendData,
1659 PCVDGEOMETRY pPCHSGeometry)
1660{
1661 RT_NOREF1(pPCHSGeometry);
1662 LogFlowFunc(("pBackendData=%#p pPCHSGeometry=%#p PCHS=%u/%u/%u\n",
1663 pBackendData, pPCHSGeometry, pPCHSGeometry->cCylinders, pPCHSGeometry->cHeads, pPCHSGeometry->cSectors));
1664 PCUEIMAGE pThis = (PCUEIMAGE)pBackendData;
1665 int rc = VINF_SUCCESS;
1666
1667 AssertPtrReturn(pThis, VERR_VD_NOT_OPENED);
1668
1669 if (pThis->uOpenFlags & VD_OPEN_FLAGS_READONLY)
1670 rc = VERR_VD_IMAGE_READ_ONLY;
1671 else
1672 rc = VERR_NOT_SUPPORTED;
1673
1674 LogFlowFunc(("returns %Rrc\n", rc));
1675 return rc;
1676}
1677
1678/** @copydoc VDIMAGEBACKEND::pfnGetLCHSGeometry */
1679static DECLCALLBACK(int) cueGetLCHSGeometry(void *pBackendData,
1680 PVDGEOMETRY pLCHSGeometry)
1681{
1682 RT_NOREF1(pLCHSGeometry);
1683 LogFlowFunc(("pBackendData=%#p pLCHSGeometry=%#p\n", pBackendData, pLCHSGeometry));
1684 PCUEIMAGE pThis = (PCUEIMAGE)pBackendData;
1685 int rc = VINF_SUCCESS;
1686
1687 AssertPtrReturn(pThis, VERR_VD_NOT_OPENED);
1688
1689 rc = VERR_NOT_SUPPORTED;
1690
1691 LogFlowFunc(("returns %Rrc (LCHS=%u/%u/%u)\n", rc, pLCHSGeometry->cCylinders, pLCHSGeometry->cHeads, pLCHSGeometry->cSectors));
1692 return rc;
1693}
1694
1695/** @copydoc VDIMAGEBACKEND::pfnSetLCHSGeometry */
1696static DECLCALLBACK(int) cueSetLCHSGeometry(void *pBackendData,
1697 PCVDGEOMETRY pLCHSGeometry)
1698{
1699 RT_NOREF1(pLCHSGeometry);
1700 LogFlowFunc(("pBackendData=%#p pLCHSGeometry=%#p LCHS=%u/%u/%u\n",
1701 pBackendData, pLCHSGeometry, pLCHSGeometry->cCylinders, pLCHSGeometry->cHeads, pLCHSGeometry->cSectors));
1702 PCUEIMAGE pThis = (PCUEIMAGE)pBackendData;
1703 int rc = VINF_SUCCESS;
1704
1705 AssertPtrReturn(pThis, VERR_VD_NOT_OPENED);
1706
1707 if (pThis->uOpenFlags & VD_OPEN_FLAGS_READONLY)
1708 rc = VERR_VD_IMAGE_READ_ONLY;
1709 else
1710 rc = VERR_NOT_SUPPORTED;
1711
1712 LogFlowFunc(("returns %Rrc\n", rc));
1713 return rc;
1714}
1715
1716/** @copydoc VDIMAGEBACKEND::pfnQueryRegions */
1717static DECLCALLBACK(int) cueQueryRegions(void *pBackendData, PCVDREGIONLIST *ppRegionList)
1718{
1719 LogFlowFunc(("pBackendData=%#p ppRegionList=%#p\n", pBackendData, ppRegionList));
1720 PCUEIMAGE pThis = (PCUEIMAGE)pBackendData;
1721
1722 AssertPtrReturn(pThis, VERR_VD_NOT_OPENED);
1723
1724 *ppRegionList = pThis->pRegionList;
1725 LogFlowFunc(("returns %Rrc\n", VINF_SUCCESS));
1726 return VINF_SUCCESS;
1727}
1728
1729/** @copydoc VDIMAGEBACKEND::pfnRegionListRelease */
1730static DECLCALLBACK(void) cueRegionListRelease(void *pBackendData, PCVDREGIONLIST pRegionList)
1731{
1732 RT_NOREF1(pRegionList);
1733 LogFlowFunc(("pBackendData=%#p pRegionList=%#p\n", pBackendData, pRegionList));
1734 PCUEIMAGE pThis = (PCUEIMAGE)pBackendData;
1735 AssertPtr(pThis); RT_NOREF(pThis);
1736
1737 /* Nothing to do here. */
1738}
1739
1740/** @copydoc VDIMAGEBACKEND::pfnGetImageFlags */
1741static DECLCALLBACK(unsigned) cueGetImageFlags(void *pBackendData)
1742{
1743 LogFlowFunc(("pBackendData=%#p\n", pBackendData));
1744 PCUEIMAGE pThis = (PCUEIMAGE)pBackendData;
1745
1746 AssertPtrReturn(pThis, 0);
1747
1748 LogFlowFunc(("returns %#x\n", pThis->uImageFlags));
1749 return pThis->uImageFlags;
1750}
1751
1752/** @copydoc VDIMAGEBACKEND::pfnGetOpenFlags */
1753static DECLCALLBACK(unsigned) cueGetOpenFlags(void *pBackendData)
1754{
1755 LogFlowFunc(("pBackendData=%#p\n", pBackendData));
1756 PCUEIMAGE pThis = (PCUEIMAGE)pBackendData;
1757
1758 AssertPtrReturn(pThis, 0);
1759
1760 LogFlowFunc(("returns %#x\n", pThis->uOpenFlags));
1761 return pThis->uOpenFlags;
1762}
1763
1764/** @copydoc VDIMAGEBACKEND::pfnSetOpenFlags */
1765static DECLCALLBACK(int) cueSetOpenFlags(void *pBackendData, unsigned uOpenFlags)
1766{
1767 LogFlowFunc(("pBackendData=%#p\n uOpenFlags=%#x", pBackendData, uOpenFlags));
1768 PCUEIMAGE pThis = (PCUEIMAGE)pBackendData;
1769 int rc = VINF_SUCCESS;
1770
1771 /* Image must be opened and the new flags must be valid. */
1772 if (!pThis || (uOpenFlags & ~( VD_OPEN_FLAGS_READONLY | VD_OPEN_FLAGS_INFO
1773 | VD_OPEN_FLAGS_ASYNC_IO | VD_OPEN_FLAGS_SHAREABLE
1774 | VD_OPEN_FLAGS_SEQUENTIAL | VD_OPEN_FLAGS_SKIP_CONSISTENCY_CHECKS)))
1775 rc = VERR_INVALID_PARAMETER;
1776 else
1777 {
1778 /* Implement this operation via reopening the image. */
1779 rc = cueFreeImage(pThis, false);
1780 if (RT_SUCCESS(rc))
1781 rc = cueOpenImage(pThis, uOpenFlags);
1782 }
1783
1784 LogFlowFunc(("returns %Rrc\n", rc));
1785 return rc;
1786}
1787
1788/** @copydoc VDIMAGEBACKEND::pfnGetComment */
1789static DECLCALLBACK(int) cueGetComment(void *pBackendData, char *pszComment,
1790 size_t cbComment)
1791{
1792 RT_NOREF2(pszComment, cbComment);
1793 LogFlowFunc(("pBackendData=%#p pszComment=%#p cbComment=%zu\n", pBackendData, pszComment, cbComment));
1794 PCUEIMAGE pThis = (PCUEIMAGE)pBackendData;
1795
1796 AssertPtrReturn(pThis, VERR_VD_NOT_OPENED);
1797
1798 LogFlowFunc(("returns %Rrc comment='%s'\n", VERR_NOT_SUPPORTED, pszComment));
1799 return VERR_NOT_SUPPORTED;
1800}
1801
1802/** @copydoc VDIMAGEBACKEND::pfnSetComment */
1803static DECLCALLBACK(int) cueSetComment(void *pBackendData, const char *pszComment)
1804{
1805 RT_NOREF1(pszComment);
1806 LogFlowFunc(("pBackendData=%#p pszComment=\"%s\"\n", pBackendData, pszComment));
1807 PCUEIMAGE pThis = (PCUEIMAGE)pBackendData;
1808
1809 AssertPtrReturn(pThis, VERR_VD_NOT_OPENED);
1810
1811 int rc;
1812 if (pThis->uOpenFlags & VD_OPEN_FLAGS_READONLY)
1813 rc = VERR_VD_IMAGE_READ_ONLY;
1814 else
1815 rc = VERR_NOT_SUPPORTED;
1816
1817 LogFlowFunc(("returns %Rrc\n", rc));
1818 return rc;
1819}
1820
1821/** @copydoc VDIMAGEBACKEND::pfnGetUuid */
1822static DECLCALLBACK(int) cueGetUuid(void *pBackendData, PRTUUID pUuid)
1823{
1824 RT_NOREF1(pUuid);
1825 LogFlowFunc(("pBackendData=%#p pUuid=%#p\n", pBackendData, pUuid));
1826 PCUEIMAGE pThis = (PCUEIMAGE)pBackendData;
1827
1828 AssertPtrReturn(pThis, VERR_VD_NOT_OPENED);
1829
1830 LogFlowFunc(("returns %Rrc (%RTuuid)\n", VERR_NOT_SUPPORTED, pUuid));
1831 return VERR_NOT_SUPPORTED;
1832}
1833
1834/** @copydoc VDIMAGEBACKEND::pfnSetUuid */
1835static DECLCALLBACK(int) cueSetUuid(void *pBackendData, PCRTUUID pUuid)
1836{
1837 RT_NOREF1(pUuid);
1838 LogFlowFunc(("pBackendData=%#p Uuid=%RTuuid\n", pBackendData, pUuid));
1839 PCUEIMAGE pThis = (PCUEIMAGE)pBackendData;
1840
1841 AssertPtrReturn(pThis, VERR_VD_NOT_OPENED);
1842
1843 int rc;
1844 if (pThis->uOpenFlags & VD_OPEN_FLAGS_READONLY)
1845 rc = VERR_VD_IMAGE_READ_ONLY;
1846 else
1847 rc = VERR_NOT_SUPPORTED;
1848
1849 LogFlowFunc(("returns %Rrc\n", rc));
1850 return rc;
1851}
1852
1853/** @copydoc VDIMAGEBACKEND::pfnGetModificationUuid */
1854static DECLCALLBACK(int) cueGetModificationUuid(void *pBackendData, PRTUUID pUuid)
1855{
1856 RT_NOREF1(pUuid);
1857 LogFlowFunc(("pBackendData=%#p pUuid=%#p\n", pBackendData, pUuid));
1858 PCUEIMAGE pThis = (PCUEIMAGE)pBackendData;
1859
1860 AssertPtrReturn(pThis, VERR_VD_NOT_OPENED);
1861
1862 LogFlowFunc(("returns %Rrc (%RTuuid)\n", VERR_NOT_SUPPORTED, pUuid));
1863 return VERR_NOT_SUPPORTED;
1864}
1865
1866/** @copydoc VDIMAGEBACKEND::pfnSetModificationUuid */
1867static DECLCALLBACK(int) cueSetModificationUuid(void *pBackendData, PCRTUUID pUuid)
1868{
1869 RT_NOREF1(pUuid);
1870 LogFlowFunc(("pBackendData=%#p Uuid=%RTuuid\n", pBackendData, pUuid));
1871 PCUEIMAGE pThis = (PCUEIMAGE)pBackendData;
1872
1873 AssertPtrReturn(pThis, VERR_VD_NOT_OPENED);
1874
1875 int rc;
1876 if (pThis->uOpenFlags & VD_OPEN_FLAGS_READONLY)
1877 rc = VERR_VD_IMAGE_READ_ONLY;
1878 else
1879 rc = VERR_NOT_SUPPORTED;
1880
1881 LogFlowFunc(("returns %Rrc\n", rc));
1882 return rc;
1883}
1884
1885/** @copydoc VDIMAGEBACKEND::pfnGetParentUuid */
1886static DECLCALLBACK(int) cueGetParentUuid(void *pBackendData, PRTUUID pUuid)
1887{
1888 RT_NOREF1(pUuid);
1889 LogFlowFunc(("pBackendData=%#p pUuid=%#p\n", pBackendData, pUuid));
1890 PCUEIMAGE pThis = (PCUEIMAGE)pBackendData;
1891
1892 AssertPtrReturn(pThis, VERR_VD_NOT_OPENED);
1893
1894 LogFlowFunc(("returns %Rrc (%RTuuid)\n", VERR_NOT_SUPPORTED, pUuid));
1895 return VERR_NOT_SUPPORTED;
1896}
1897
1898/** @copydoc VDIMAGEBACKEND::pfnSetParentUuid */
1899static DECLCALLBACK(int) cueSetParentUuid(void *pBackendData, PCRTUUID pUuid)
1900{
1901 RT_NOREF1(pUuid);
1902 LogFlowFunc(("pBackendData=%#p Uuid=%RTuuid\n", pBackendData, pUuid));
1903 PCUEIMAGE pThis = (PCUEIMAGE)pBackendData;
1904
1905 AssertPtrReturn(pThis, VERR_VD_NOT_OPENED);
1906
1907 int rc;
1908 if (pThis->uOpenFlags & VD_OPEN_FLAGS_READONLY)
1909 rc = VERR_VD_IMAGE_READ_ONLY;
1910 else
1911 rc = VERR_NOT_SUPPORTED;
1912
1913 LogFlowFunc(("returns %Rrc\n", rc));
1914 return rc;
1915}
1916
1917/** @copydoc VDIMAGEBACKEND::pfnGetParentModificationUuid */
1918static DECLCALLBACK(int) cueGetParentModificationUuid(void *pBackendData, PRTUUID pUuid)
1919{
1920 RT_NOREF1(pUuid);
1921 LogFlowFunc(("pBackendData=%#p pUuid=%#p\n", pBackendData, pUuid));
1922 PCUEIMAGE pThis = (PCUEIMAGE)pBackendData;
1923
1924 AssertPtrReturn(pThis, VERR_VD_NOT_OPENED);
1925
1926 int rc;
1927 if (pThis->uOpenFlags & VD_OPEN_FLAGS_READONLY)
1928 rc = VERR_VD_IMAGE_READ_ONLY;
1929 else
1930 rc = VERR_NOT_SUPPORTED;
1931
1932 LogFlowFunc(("returns %Rrc (%RTuuid)\n", rc, pUuid));
1933 return rc;
1934}
1935
1936/** @copydoc VDIMAGEBACKEND::pfnSetParentModificationUuid */
1937static DECLCALLBACK(int) cueSetParentModificationUuid(void *pBackendData, PCRTUUID pUuid)
1938{
1939 RT_NOREF1(pUuid);
1940 LogFlowFunc(("pBackendData=%#p Uuid=%RTuuid\n", pBackendData, pUuid));
1941 PCUEIMAGE pThis = (PCUEIMAGE)pBackendData;
1942
1943 AssertPtrReturn(pThis, VERR_VD_NOT_OPENED);
1944
1945 int rc;
1946 if (pThis->uOpenFlags & VD_OPEN_FLAGS_READONLY)
1947 rc = VERR_VD_IMAGE_READ_ONLY;
1948 else
1949 rc = VERR_NOT_SUPPORTED;
1950
1951 LogFlowFunc(("returns %Rrc\n", rc));
1952 return rc;
1953}
1954
1955/** @copydoc VDIMAGEBACKEND::pfnDump */
1956static DECLCALLBACK(void) cueDump(void *pBackendData)
1957{
1958 PCUEIMAGE pThis = (PCUEIMAGE)pBackendData;
1959
1960 AssertPtrReturnVoid(pThis);
1961 vdIfErrorMessage(pThis->pIfError, "Dumping CUE image \"%s\" mode=%s uOpenFlags=%X File=%#p\n",
1962 pThis->pszFilename,
1963 (pThis->uOpenFlags & VD_OPEN_FLAGS_READONLY) ? "r/o" : "r/w",
1964 pThis->uOpenFlags,
1965 pThis->pStorage);
1966 vdIfErrorMessage(pThis->pIfError, "Backing File \"%s\" File=%#p\n",
1967 pThis->pszDataFilename, pThis->pStorageData);
1968 vdIfErrorMessage(pThis->pIfError, "Number of tracks: %u\n", pThis->pRegionList->cRegions);
1969 for (uint32_t i = 0; i < pThis->pRegionList->cRegions; i++)
1970 {
1971 PCVDREGIONDESC pRegion = &pThis->pRegionList->aRegions[i];
1972
1973 vdIfErrorMessage(pThis->pIfError, "------------------------ Track %u ------------------------\n", i);
1974 vdIfErrorMessage(pThis->pIfError, "Start=%llu Size=%llu BlockSize=%llu DataSize=%llu MetadataSize=%llu\n",
1975 pRegion->offRegion, pRegion->cRegionBlocksOrBytes, pRegion->cbBlock, pRegion->cbData,
1976 pRegion->cbMetadata);
1977 vdIfErrorMessage(pThis->pIfError, "DataForm=%s MetadataForm=%s\n",
1978 cueRegionDataFormStringify(pRegion->enmDataForm),
1979 cueRegionMetadataFormStringify(pRegion->enmMetadataForm));
1980 }
1981}
1982
1983
1984
1985const VDIMAGEBACKEND g_CueBackend =
1986{
1987 /* u32Version */
1988 VD_IMGBACKEND_VERSION,
1989 /* pszBackendName */
1990 "CUE",
1991 /* uBackendCaps */
1992 VD_CAP_FILE | VD_CAP_VFS,
1993 /* paFileExtensions */
1994 s_aCueFileExtensions,
1995 /* paConfigInfo */
1996 NULL,
1997 /* pfnProbe */
1998 cueProbe,
1999 /* pfnOpen */
2000 cueOpen,
2001 /* pfnCreate */
2002 NULL,
2003 /* pfnRename */
2004 NULL,
2005 /* pfnClose */
2006 cueClose,
2007 /* pfnRead */
2008 cueRead,
2009 /* pfnWrite */
2010 cueWrite,
2011 /* pfnFlush */
2012 cueFlush,
2013 /* pfnDiscard */
2014 NULL,
2015 /* pfnGetVersion */
2016 cueGetVersion,
2017 /* pfnGetSectorSize */
2018 cueGetSectorSize,
2019 /* pfnGetSize */
2020 cueGetSize,
2021 /* pfnGetFileSize */
2022 cueGetFileSize,
2023 /* pfnGetPCHSGeometry */
2024 cueGetPCHSGeometry,
2025 /* pfnSetPCHSGeometry */
2026 cueSetPCHSGeometry,
2027 /* pfnGetLCHSGeometry */
2028 cueGetLCHSGeometry,
2029 /* pfnSetLCHSGeometry */
2030 cueSetLCHSGeometry,
2031 /* pfnQueryRegions */
2032 cueQueryRegions,
2033 /* pfnRegionListRelease */
2034 cueRegionListRelease,
2035 /* pfnGetImageFlags */
2036 cueGetImageFlags,
2037 /* pfnGetOpenFlags */
2038 cueGetOpenFlags,
2039 /* pfnSetOpenFlags */
2040 cueSetOpenFlags,
2041 /* pfnGetComment */
2042 cueGetComment,
2043 /* pfnSetComment */
2044 cueSetComment,
2045 /* pfnGetUuid */
2046 cueGetUuid,
2047 /* pfnSetUuid */
2048 cueSetUuid,
2049 /* pfnGetModificationUuid */
2050 cueGetModificationUuid,
2051 /* pfnSetModificationUuid */
2052 cueSetModificationUuid,
2053 /* pfnGetParentUuid */
2054 cueGetParentUuid,
2055 /* pfnSetParentUuid */
2056 cueSetParentUuid,
2057 /* pfnGetParentModificationUuid */
2058 cueGetParentModificationUuid,
2059 /* pfnSetParentModificationUuid */
2060 cueSetParentModificationUuid,
2061 /* pfnDump */
2062 cueDump,
2063 /* pfnGetTimestamp */
2064 NULL,
2065 /* pfnGetParentTimestamp */
2066 NULL,
2067 /* pfnSetParentTimestamp */
2068 NULL,
2069 /* pfnGetParentFilename */
2070 NULL,
2071 /* pfnSetParentFilename */
2072 NULL,
2073 /* pfnComposeLocation */
2074 genericFileComposeLocation,
2075 /* pfnComposeName */
2076 genericFileComposeName,
2077 /* pfnCompact */
2078 NULL,
2079 /* pfnResize */
2080 NULL,
2081 /* pfnRepair */
2082 NULL,
2083 /* pfnTraverseMetadata */
2084 NULL,
2085 /* u32VersionEnd */
2086 VD_IMGBACKEND_VERSION
2087};
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette