This file is indexed.

/usr/include/exempi-2.0/exempi/xmp.h is in libexempi-dev 2.4.1-1.

This file is owned by root:root, with mode 0o644.

The actual contents of the file can be viewed below.

  1
  2
  3
  4
  5
  6
  7
  8
  9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 43
 44
 45
 46
 47
 48
 49
 50
 51
 52
 53
 54
 55
 56
 57
 58
 59
 60
 61
 62
 63
 64
 65
 66
 67
 68
 69
 70
 71
 72
 73
 74
 75
 76
 77
 78
 79
 80
 81
 82
 83
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
/*
 * exempi - xmp.h
 *
 * Copyright (C) 2007-2016 Hubert Figuiere
 * Copyright 2002-2007 Adobe Systems Incorporated
 * All rights reserved.
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions
 * are met:
 *
 * 1 Redistributions of source code must retain the above copyright
 * notice, this list of conditions and the following disclaimer.
 *
 * 2 Redistributions in binary form must reproduce the above copyright
 * notice, this list of conditions and the following disclaimer in the
 * documentation and/or other materials provided with the
 * distribution.
 *
 * 3 Neither the name of the Authors, nor the names of its
 * contributors may be used to endorse or promote products derived
 * from this software wit hout specific prior written permission.
 *
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
 * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
 * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
 * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
 * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
 * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED
 * OF THE POSSIBILITY OF SUCH DAMAGE.
 */

#ifndef __EXEMPI_XMP_H_
#define __EXEMPI_XMP_H_

#include <stdlib.h>
/* stdbool choke on Sun (bug #14612) */
#if !(defined(__sun) && defined(__cplusplus))
#include <stdbool.h>
#endif
#include <stdint.h>

#include <time.h>

#ifdef __cplusplus
extern "C" {
#endif

/* enums grafted from XMP_Const.h */

/** Option bits for xmp_files_open() */
typedef enum {
    XMP_OPEN_NOOPTION = 0x00000000,        /**< No open option */
    XMP_OPEN_READ = 0x00000001,            /**< Open for read-only access. */
    XMP_OPEN_FORUPDATE = 0x00000002,       /**< Open for reading and writing. */
    XMP_OPEN_ONLYXMP = 0x00000004,         /**< Only the XMP is wanted,
                                            * allows space/time optimizations. */
    XMP_OPEN_CACHETNAIL = 0x00000008,      /**< Cache thumbnail if possible,
                                            * GetThumbnail will be called. */
    XMP_OPEN_STRICTLY = 0x00000010,        /**< Be strict about locating XMP
                                            * and reconciling with other forms. */
    XMP_OPEN_USESMARTHANDLER = 0x00000020, /**< Require the use of a smart
                                            * handler. */
    XMP_OPEN_USEPACKETSCANNING = 0x00000040, /**< Force packet scanning,
                                              * don't use a smart handler. */
    XMP_OPEN_LIMITSCANNING = 0x00000080,     /**< Only packet scan files "known"
                                              * to need scanning. */
    XMP_OPEN_REPAIR_FILE =
        0x00000100, /**< Attempt to repair a file opened for update,
                     * default is to not open (throw an exception). */
    XMP_OPEN_OPTIMIZEFILELAYOUT =
        0x00000200, /**< Optimize MPEG4 to support stream when updating
                     * This can take some time */
    XMP_OPEN_INBACKGROUND = 0x10000000 /**< Set if calling from background
                                        * thread. */
} XmpOpenFileOptions;

/** Option bits for xmp_files_close() */
typedef enum {
    XMP_CLOSE_NOOPTION = 0x0000,  /**< No close option */
    XMP_CLOSE_SAFEUPDATE = 0x0001 /**< Write into a temporary file and
                                   * swap for crash safety. */
} XmpCloseFileOptions;

typedef enum {

    /* Public file formats. Hex used to avoid gcc warnings. */
    /* ! Leave them as big endian. There seems to be no decent way on UNIX to
       determine the target */
    /* ! endianness at compile time. Forcing it on the client isn't acceptable.
       */

    XMP_FT_PDF = 0x50444620UL, /* 'PDF ' */
    XMP_FT_PS = 0x50532020UL,  /* 'PS  ', general PostScript following DSC
                                  conventions. */
    XMP_FT_EPS = 0x45505320UL, /* 'EPS ', encapsulated PostScript. */

    XMP_FT_JPEG = 0x4A504547UL,   /* 'JPEG' */
    XMP_FT_JPEG2K = 0x4A505820UL, /* 'JPX ', ISO 15444-1 */
    XMP_FT_TIFF = 0x54494646UL,   /* 'TIFF' */
    XMP_FT_GIF = 0x47494620UL,    /* 'GIF ' */
    XMP_FT_PNG = 0x504E4720UL,    /* 'PNG ' */
    XMP_FT_WEBP = 0x57454250UL,   /* 'WEBP' */

    XMP_FT_SWF = 0x53574620UL, /* 'SWF ' */
    XMP_FT_FLA = 0x464C4120UL, /* 'FLA ' */
    XMP_FT_FLV = 0x464C5620UL, /* 'FLV ' */

    XMP_FT_MOV = 0x4D4F5620UL,   /* 'MOV ', Quicktime */
    XMP_FT_AVI = 0x41564920UL,   /* 'AVI ' */
    XMP_FT_CIN = 0x43494E20UL,   /* 'CIN ', Cineon */
    XMP_FT_WAV = 0x57415620UL,   /* 'WAV ' */
    XMP_FT_MP3 = 0x4D503320UL,   /* 'MP3 ' */
    XMP_FT_SES = 0x53455320UL,   /* 'SES ', Audition session */
    XMP_FT_CEL = 0x43454C20UL,   /* 'CEL ', Audition loop */
    XMP_FT_MPEG = 0x4D504547UL,  /* 'MPEG' */
    XMP_FT_MPEG2 = 0x4D503220UL, /* 'MP2 ' */
    XMP_FT_MPEG4 = 0x4D503420UL, /* 'MP4 ', ISO 14494-12 and -14 */
    XMP_FT_WMAV = 0x574D4156UL,  /* 'WMAV', Windows Media Audio and Video */
    XMP_FT_AIFF = 0x41494646UL,  /* 'AIFF' */

    XMP_FT_HTML = 0x48544D4CUL, /* 'HTML' */
    XMP_FT_XML = 0x584D4C20UL,  /* 'XML ' */
    XMP_FT_TEXT = 0x74657874UL, /* 'text' */

    /* Adobe application file formats. */
    XMP_FT_PHOTOSHOP = 0x50534420UL,   /* 'PSD ' */
    XMP_FT_ILLUSTRATOR = 0x41492020UL, /* 'AI  ' */
    XMP_FT_INDESIGN = 0x494E4444UL,    /* 'INDD' */
    XMP_FT_AEPROJECT = 0x41455020UL,   /* 'AEP ' */
    XMP_FT_AEPROJTEMPLATE =
        0x41455420UL, /* 'AET ', After Effects Project Template */
    XMP_FT_AEFILTERPRESET = 0x46465820UL,  /* 'FFX ' */
    XMP_FT_ENCOREPROJECT = 0x4E434F52UL,   /* 'NCOR' */
    XMP_FT_PREMIEREPROJECT = 0x5052504AUL, /* 'PRPJ' */
    XMP_FT_PREMIERETITLE = 0x5052544CUL,   /* 'PRTL' */

    /* Catch all. */
    XMP_FT_UNKNOWN = 0x20202020UL /* '    ' */
} XmpFileType;

typedef enum {

    XMP_FMT_CAN_INJECT_XMP = 0x00000001,
    XMP_FMT_CAN_EXPAND = 0x00000002,
    XMP_FMT_CAN_REWRITE = 0x00000004,
    XMP_FMT_PREFERS_IN_PLACE = 0x00000008,
    XMP_FMT_CAN_RECONCILE = 0x00000010,
    XMP_FMT_ALLOWS_ONLY_XMP = 0x00000020,
    XMP_FMT_RETURNS_RAW_PACKET = 0x00000040,
    XMP_FMT_HANDLER_OWNS_FILE = 0x00000100,
    XMP_FMT_ALLOW_SAFE_UPDATE = 0x00000200,
    XMP_FMT_NEEDS_READONLY_PACKET = 0x00000400,
    XMP_FMT_USE_SIDECAR_XMP = 0x00000800,
    XMP_FMT_FOLDER_BASED_FORMAT = 0x00001000,

    _XMP_FMT_LAST
} XmpFileFormatOptions;

typedef enum {
    XMP_ITER_CLASSMASK = 0x00FFUL,  /**< The low 8 bits are an enum of
                                     * what data structure to iterate. */
    XMP_ITER_PROPERTIES = 0x0000UL, /**< Iterate the property tree of
                                     * a TXMPMeta object. */
    XMP_ITER_ALIASES = 0x0001UL,    /**< Iterate the global alias table. */
    XMP_ITER_NAMESPACES = 0x0002UL, /**< Iterate the global namespace table. */
    XMP_ITER_JUSTCHILDREN = 0x0100UL,   /**< Just do the immediate children
                                         * of the root, default is subtree. */
    XMP_ITER_JUSTLEAFNODES = 0x0200UL,  /**< Just do the leaf nodes, default
                                         * is all nodes in the subtree. */
    XMP_ITER_JUSTLEAFNAME = 0x0400UL,   /**< Return just the leaf part of the
                                         * path, default is the full path. */
    XMP_ITER_INCLUDEALIASES = 0x0800UL, /**< Include aliases, default is just
                                         * actual properties. */
    XMP_ITER_OMITQUALIFIERS = 0x1000UL  /* Omit all qualifiers. */
} XmpIterOptions;

typedef enum {
    XMP_ITER_SKIPSUBTREE = 0x0001UL, /**< Skip the subtree below the
                                      * current node. */
    XMP_ITER_SKIPSIBLINGS =
        0x0002UL /**< Skip the subtree below and remaining
                                                      * siblings of the current
                  * node. */
} XmpIterSkipOptions;

typedef enum {
    /** Options relating to the XML string form of the property value. */
    XMP_PROP_VALUE_IS_URI = 0x00000002UL, /**< The value is a URI, use
                                           * rdf:resource attribute.
                                           * DISCOURAGED */
    /** Options relating to qualifiers attached to a property. */
    XMP_PROP_HAS_QUALIFIERS = 0x00000010UL, /**< The property has qualifiers,
                                             * includes rdf:type and
                                             * xml:lang. */
    XMP_PROP_IS_QUALIFIER = 0x00000020UL,   /**< This is a qualifier,
                                             * includes rdf:type and
                                             * xml:lang. */
    XMP_PROP_HAS_LANG = 0x00000040UL,       /**< Implies XMP_PropHasQualifiers,
                                             * property has xml:lang. */
    XMP_PROP_HAS_TYPE = 0x00000080UL,       /**< Implies XMP_PropHasQualifiers,
                                             * property has rdf:type. */

    /* Options relating to the data structure form. */
    XMP_PROP_VALUE_IS_STRUCT = 0x00000100UL, /**< The value is a structure
                                              * with nested fields. */
    XMP_PROP_VALUE_IS_ARRAY = 0x00000200UL,  /**< The value is an array
                                              * (RDF alt/bag/seq). */
    XMP_PROP_ARRAY_IS_UNORDERED = XMP_PROP_VALUE_IS_ARRAY, /**< The item order
                                                            * does not matter.*/
    XMP_PROP_ARRAY_IS_ORDERED = 0x00000400UL, /**< Implies XMP_PropValueIsArray,
                                               * item order matters. */
    XMP_PROP_ARRAY_IS_ALT = 0x00000800UL, /**< Implies XMP_PropArrayIsOrdered,
                                           * items are alternates. */

    /** Additional struct and array options. */
    XMP_PROP_ARRAY_IS_ALTTEXT =
        0x00001000UL, /**< Implies kXMP_PropArrayIsAlternate,
                       * items are localized text. */
    XMP_PROP_ARRAY_INSERT_BEFORE =
        0x00004000UL,                           /**< Used by array functions. */
    XMP_PROP_ARRAY_INSERT_AFTER = 0x00008000UL, /**< Used by array functions. */

    /* Other miscellaneous options. */
    XMP_PROP_IS_ALIAS = 0x00010000UL,    /**< This property is an alias name for
                                            another property. */
    XMP_PROP_HAS_ALIASES = 0x00020000UL, /**< This property is the base value
                                            for a set of aliases. */
    XMP_PROP_IS_INTERNAL = 0x00040000UL, /**< This property is an "internal"
                                            property, owned by applications. */
    XMP_PROP_IS_STABLE = 0x00100000UL,   /**< This property is not derived from
                                          * the document content.
                                            */
    XMP_PROP_IS_DERIVED = 0x00200000UL,  /**< This property is derived from the
                                            document content. */
    /* kXMPUtil_AllowCommas   = 0x10000000UL,  ! Used by
       TXMPUtils::CatenateArrayItems and ::SeparateArrayItems. */
    /* kXMP_DeleteExisting    = 0x20000000UL,  ! Used by TXMPMeta::SetXyz
       functions to delete any pre-existing property. */
    /* kXMP_SchemaNode        = 0x80000000UL,  ! Returned by iterators - #define
       to avoid warnings */

    /* Masks that are multiple flags. */
    XMP_PROP_ARRAY_FORM_MASK = XMP_PROP_VALUE_IS_ARRAY |
                               XMP_PROP_ARRAY_IS_ORDERED |
                               XMP_PROP_ARRAY_IS_ALT |
                               XMP_PROP_ARRAY_IS_ALTTEXT,
    XMP_PROP_COMPOSITE_MASK = XMP_PROP_VALUE_IS_STRUCT |
                              XMP_PROP_ARRAY_FORM_MASK, /* Is it simple or
                                                           composite (array or
                                                           struct)? */
    XMP_IMPL_RESERVED_MASK =
        0x70000000L /**< Reserved for transient use by the implementation. */
} XmpPropsBits;

/* convenience macros */

/** set option bits */
#define XMP_SET_OPTION(var, opt) var |= (opt)
/** clear option bits */
#define XMP_CLEAR_OPTION(var, opt) var &= ~(opt)
/** test if option is set */
#define XMP_TEST_OPTION_SET(var, opt) (((var) & (opt)) != 0)
/** test if option is clear */
#define XMP_TEST_OPTION_CLEAR(var, opt) (((var) & (opt)) == 0)

#define XMP_IS_PROP_SIMPLE(opt) (((opt)&XMP_PROP_COMPOSITE_MASK) == 0)
#define XMP_IS_PROP_STRUCT(opt) (((opt)&XMP_PROP_VALUE_IS_STRUCT) != 0)
#define XMP_IS_PROP_ARRAY(opt) (((opt)&XMP_PROP_VALUE_IS_ARRAY) != 0)
#define XMP_IS_ARRAY_UNORDERED(opt) (((opt)&XMP_PROP_ARRAY_IS_ORDERED) == 0)
#define XMP_IS_ARRAY_ORDERED(opt) (((opt)&XMP_PROP_ARRAY_IS_ORDERED) != 0)
#define XMP_IS_ARRAY_ALT(opt) (((opt)&XMP_PROP_ARRAY_IS_ALT) != 0)
#define XMP_IS_ARRAY_ALTTEXT(opt) (((opt)&XMP_PROP_ARRAY_IS_ALTTEXT) != 0)

#define XMP_HAS_PROP_QUALIFIERS(opt) (((opt)&XMP_PROP_HAS_QUALIFIERS) != 0)
#define XMP_IS_PROP_QUALIFIER(opt) (((opt)&XMP_PROP_IS_QUALIFIER) != 0)
#define XMP_HAS_PROP_LANG(opt) (((opt)&XMP_PROP_HAS_LANG) != 0)

#define XMP_IS_NODE_SCHEMA(opt) (((opt)&XMP_SCHEMA_NODE) != 0)
#define XMP_IS_PROP_ALIAS(opt) (((opt)&XMP_PROP_IS_ALIAS) != 0)

enum {                                          /* Options for xmp_serialize */
       XMP_SERIAL_OMITPACKETWRAPPER = 0x0010UL, /**< Omit the XML packet
                                                 * wrapper. */
       XMP_SERIAL_READONLYPACKET = 0x0020UL,    /**< Default is a writeable
                                                 * packet. */
       XMP_SERIAL_USECOMPACTFORMAT = 0x0040UL,  /**< Use a compact form of
                                                 * RDF. */

       XMP_SERIAL_INCLUDETHUMBNAILPAD =
           0x0100UL,                            /**< Include a padding allowance
                                                 * for a thumbnail image. */
       XMP_SERIAL_EXACTPACKETLENGTH = 0x0200UL, /**< The padding parameter is
                                                 * the overall packet length. */
       XMP_SERIAL_WRITEALIASCOMMENTS = 0x0400UL, /**< Show aliases as XML
                                                  * comments. */
       XMP_SERIAL_OMITALLFORMATTING = 0x0800UL,  /**< Omit all formatting
                                                  * whitespace. */

       _XMP_LITTLEENDIAN_BIT =
           0x0001UL, /* ! Don't use directly, see the combined values below! */
       _XMP_UTF16_BIT = 0x0002UL,
       _XMP_UTF32_BIT = 0x0004UL,

       XMP_SERIAL_ENCODINGMASK = 0x0007UL,
       XMP_SERIAL_ENCODEUTF8 = 0UL,
       XMP_SERIAL_ENCODEUTF16BIG = _XMP_UTF16_BIT,
       XMP_SERIAL_ENCODEUTF16LITTLE = _XMP_UTF16_BIT | _XMP_LITTLEENDIAN_BIT,
       XMP_SERIAL_ENCODEUTF32BIG = _XMP_UTF32_BIT,
       XMP_SERIAL_ENCODEUTF32LITTLE = _XMP_UTF32_BIT | _XMP_LITTLEENDIAN_BIT
};

/** pointer to XMP packet. Opaque. */
typedef struct _Xmp *XmpPtr;
typedef struct _XmpFile *XmpFilePtr;
typedef struct _XmpString *XmpStringPtr;
typedef struct _XmpIterator *XmpIteratorPtr;

typedef struct _XmpDateTime {
    int32_t year;
    int32_t month;    /* 1..12 */
    int32_t day;      /* 1..31 */
    int32_t hour;     /* 0..23 */
    int32_t minute;   /* 0..59 */
    int32_t second;   /* 0..59 */
    int32_t tzSign;   /* -1..+1, 0 means UTC, -1 is west, +1 is east. */
    int32_t tzHour;   /* 0..23 */
    int32_t tzMinute; /* 0..59 */
    int32_t nanoSecond;
} XmpDateTime;

typedef struct _XmpPacketInfo {
  /// Packet offset in the file in bytes, -1 if unknown.
  int64_t offset;
  /// Packet length in the file in bytes, -1 if unknown.
  int32_t length;
  /// Packet padding size in bytes, zero if unknown.
  int32_t padSize;

  /// Character format using the values \c kXMP_Char8Bit,
  /// \c kXMP_Char16BitBig, etc.
  uint8_t  charForm;
  /// True if there is a packet wrapper and the trailer says writeable
  /// by dumb packet scanners.
  bool  writeable;
  /// True if there is a packet wrapper, the "<?xpacket...>"
  /// XML processing instructions.
  bool  hasWrapper;

  /// Padding to make the struct's size be a multiple 4.
  uint8_t  pad;
} XmpPacketInfo;

/** Values used for tzSign field. */
enum {
    XMP_TZ_WEST = -1, /**< West of UTC   */
    XMP_TZ_UTC = 0,   /**< UTC           */
    XMP_TZ_EAST = +1  /**< East of UTC   */
};

/** Init the library. Must be called before anything else */
bool xmp_init(void);
void xmp_terminate(void);

/** get the error code that last occurred.
 * @todo make this thread-safe. Getting the error code
 * from another thread than the on it occurred in is undefined.
 */
int xmp_get_error(void);

XmpFilePtr xmp_files_new(void);
XmpFilePtr xmp_files_open_new(const char *, XmpOpenFileOptions options);

bool xmp_files_open(XmpFilePtr xf, const char *, XmpOpenFileOptions options);

/** Close an XMP file. Will flush the changes
 * @param xf the file object
 * @param options the options to close.
 * @return true on succes, false on error
 * xmp_get_error() will give the error code.
 */
bool xmp_files_close(XmpFilePtr xf, XmpCloseFileOptions options);

/** Get the XMP packet from the file
 * @param xf the %XmpFilePtr to get the XMP packet from
 * @return a newly allocated XmpPtr. Use %xmp_free to release it.
 */
XmpPtr xmp_files_get_new_xmp(XmpFilePtr xf);

/** File the XMP packet from the file
 * @param xf the %XmpFilePtr to get the XMP packet from
 * @param xmp the XMP Packet to fill. Must be valid.
 */
bool xmp_files_get_xmp(XmpFilePtr xf, XmpPtr xmp);
bool xmp_files_get_xmp_xmpstring(XmpFilePtr xf, XmpStringPtr xmp_packet,
                                 XmpPacketInfo* packet_info);

bool xmp_files_can_put_xmp(XmpFilePtr xf, XmpPtr xmp);
bool xmp_files_can_put_xmp_xmpstring(XmpFilePtr xf, XmpStringPtr xmp_packet);
bool xmp_files_can_put_xmp_cstr(XmpFilePtr xf, const char* xmp_packet,
                                size_t len);

bool xmp_files_put_xmp(XmpFilePtr xf, XmpPtr xmp);
bool xmp_files_put_xmp_xmpstring(XmpFilePtr xf, XmpStringPtr xmp_packet);
bool xmp_files_put_xmp_cstr(XmpFilePtr xf, const char* xmp_packet, size_t len);

/** Get the file info from the open file
 * @param xf the file object
 * @param[out] filePath the file path object to store the path in. Pass NULL if
 * not needed.
 * @param[out] options the options for open. Pass NULL if not needed.
 * @param[out] file_format the detected file format. Pass NULL if not needed.
 * @param[out] handler_flags the format options like from
 * %xmp_files_get_format_info.
 * @return false in case of error.
 */
bool xmp_files_get_file_info(XmpFilePtr xf, XmpStringPtr filePath,
                             XmpOpenFileOptions *options,
                             XmpFileType *file_format,
                             XmpFileFormatOptions *handler_flags);

/** Free a XmpFilePtr
 * @param xf the file ptr. Cannot be NULL
 * @return false on error.
 * Calll %xmp_get_error to retrieve the error code.
 */
bool xmp_files_free(XmpFilePtr xf);

/** Get the format info
 * @param format type identifier
 * @param options the options for the file format handler
 * @return false on error
 */
bool xmp_files_get_format_info(XmpFileType format,
                               XmpFileFormatOptions *options);

/** Check the file format of a file. Use the same logic as in xmp_files_open()
 * @param filePath the path to the file
 * @return XMP_FT_UNKNOWN on error or if the file type is unknown
 */
XmpFileType xmp_files_check_file_format(const char *filePath);

/** Register a new namespace to add properties to
 *  This is done automatically when reading the metadata block
 *  @param namespaceURI the namespace URI to register
 *  @param suggestedPrefix the suggested prefix
 *  @param registeredPrefix the really registered prefix. Not necessarily
 *  %suggestedPrefix.
 *  @return true if success, false otherwise.
 */
bool xmp_register_namespace(const char *namespaceURI,
                            const char *suggestedPrefix,
                            XmpStringPtr registeredPrefix);

/** Check is a namespace is registered
 *  @param ns the namespace to check.
 *  @param prefix The prefix associated if registered. Pass NULL
 *  if not interested.
 *  @return true if registered.
 *  NEW in 2.1
 */
bool xmp_namespace_prefix(const char *ns, XmpStringPtr prefix);

/** Check if a ns prefix is registered.
 *  @param prefix the prefix to check.
 *  @param ns the namespace associated if registered. Pass NULL
 *  if not interested.
 *  @return true if registered.
 *  NEW in 2.1
 */
bool xmp_prefix_namespace_uri(const char *prefix, XmpStringPtr ns);

/** Create a new empty XMP packet
 * @return the packet pointer. Must be free with xmp_free()
 */
XmpPtr xmp_new_empty(void);

/** Create a new XMP packet
 * @param buffer the buffer to load data from. UTF-8 encoded.
 * @param len the buffer length in byte
 * @return the packet pointer. Must be free with xmp_free()
 */
XmpPtr xmp_new(const char *buffer, size_t len);

/** Create a new XMP packet from the one passed.
 * @param xmp the instance to copy. Can be NULL.
 * @return the packet pointer. NULL is failer (or NULL is passed).
 */
XmpPtr xmp_copy(XmpPtr xmp);

/** Free the xmp packet
 * @param xmp the xmp packet to free
 */
bool xmp_free(XmpPtr xmp);

/** Parse the XML passed through the buffer and load it.
 * @param xmp the XMP packet.
 * @param buffer the buffer.
 * @param len the length of the buffer.
 */
bool xmp_parse(XmpPtr xmp, const char *buffer, size_t len);

/** Serialize the XMP Packet to the given buffer
 * @param xmp the XMP Packet
 * @param buffer the buffer to write the XMP to
 * @param options options on how to write the XMP.  See XMP_SERIAL_*
 * @param padding number of bytes of padding, useful for modifying
 *                embedded XMP in place.
 * @return TRUE if success.
 */
bool xmp_serialize(XmpPtr xmp, XmpStringPtr buffer, uint32_t options,
                   uint32_t padding);

/** Serialize the XMP Packet to the given buffer with formatting
 * @param xmp the XMP Packet
 * @param buffer the buffer to write the XMP to
 * @param options options on how to write the XMP.  See XMP_SERIAL_*
 * @param padding number of bytes of padding, useful for modifying
 *                embedded XMP in place.
 * @param newline the new line character to use
 * @param tab the indentation character to use
 * @param indent the initial indentation level
 * @return TRUE if success.
 */
bool xmp_serialize_and_format(XmpPtr xmp, XmpStringPtr buffer, uint32_t options,
                              uint32_t padding, const char *newline,
                              const char *tab, int32_t indent);

/** Get an XMP property and it option bits from the XMP packet
 * @param xmp the XMP packet
 * @param schema
 * @param name
 * @param property the allocated XmpStringPtr
 * @param propsBits pointer to the option bits. Pass NULL if not needed
 * @return true if found
 */
bool xmp_get_property(XmpPtr xmp, const char *schema, const char *name,
                      XmpStringPtr property, uint32_t *propsBits);

bool xmp_get_property_date(XmpPtr xmp, const char *schema, const char *name,
                           XmpDateTime *property, uint32_t *propsBits);
bool xmp_get_property_float(XmpPtr xmp, const char *schema, const char *name,
                            double *property, uint32_t *propsBits);
bool xmp_get_property_bool(XmpPtr xmp, const char *schema, const char *name,
                           bool *property, uint32_t *propsBits);
bool xmp_get_property_int32(XmpPtr xmp, const char *schema, const char *name,
                            int32_t *property, uint32_t *propsBits);
bool xmp_get_property_int64(XmpPtr xmp, const char *schema, const char *name,
                            int64_t *property, uint32_t *propsBits);

/** Get an item frpm an array property
 * @param xmp the xmp meta
 * @param schema the schema
 * @param name the property name
 * @param index the index in the array
 * @param property the property value
 * @param propsBits the property bits. Pass NULL is unwanted.
 * @return TRUE if success.
 */
bool xmp_get_array_item(XmpPtr xmp, const char *schema, const char *name,
                        int32_t index, XmpStringPtr property,
                        uint32_t *propsBits);

/** Set an XMP property in the XMP packet
 * @param xmp the XMP packet
 * @param schema
 * @param name
 * @param value 0 terminated string
 * @param optionBits
 * @return false if failure
 */
bool xmp_set_property(XmpPtr xmp, const char *schema, const char *name,
                      const char *value, uint32_t optionBits);

/** Set a date XMP property in the XMP packet
 * @param xmp the XMP packet
 * @param schema
 * @param name
 * @param value the date-time struct
 * @param optionBits
 * @return false if failure
 */
bool xmp_set_property_date(XmpPtr xmp, const char *schema, const char *name,
                           const XmpDateTime *value, uint32_t optionBits);

/** Set a float XMP property in the XMP packet
 * @param xmp the XMP packet
 * @param schema
 * @param name
 * @param value the float value
 * @param optionBits
 * @return false if failure
 */
bool xmp_set_property_float(XmpPtr xmp, const char *schema, const char *name,
                            double value, uint32_t optionBits);
bool xmp_set_property_bool(XmpPtr xmp, const char *schema, const char *name,
                           bool value, uint32_t optionBits);
bool xmp_set_property_int32(XmpPtr xmp, const char *schema, const char *name,
                            int32_t value, uint32_t optionBits);
bool xmp_set_property_int64(XmpPtr xmp, const char *schema, const char *name,
                            int64_t value, uint32_t optionBits);

bool xmp_set_array_item(XmpPtr xmp, const char *schema, const char *name,
                        int32_t index, const char *value, uint32_t optionBits);

/** Append a value to the XMP Property array in the XMP Packet provided
 * @param xmp the XMP packet
 * @param schema the schema of the property
 * @param name the name of the property
 * @param arrayOptions option bits of the parent array
 * @param value null-terminated string
 * @param optionBits option bits of the value itself.
 */
bool xmp_append_array_item(XmpPtr xmp, const char *schema, const char *name,
                           uint32_t arrayOptions, const char *value,
                           uint32_t optionBits);

/** Delete a property from the XMP Packet provided
 * @param xmp the XMP packet
 * @param schema the schema of the property
 * @param name the name of the property
 */
bool xmp_delete_property(XmpPtr xmp, const char *schema, const char *name);

/** Determines if a property exists in the XMP Packet provided
 * @param xmp the XMP packet
 * @param schema the schema of the property. Can't be NULL or empty.
 * @param name the name of the property. Can't be NULL or empty.
 * @return true is the property exists
 */
bool xmp_has_property(XmpPtr xmp, const char *schema, const char *name);

/** Get a localised text from a localisable property.
 * @param xmp the XMP packet
 * @param schema the schema
 * @param name the property name.
 * @param genericLang the generic language you may want as a fall back.
 * Can be NULL or empty.
 * @param specificLang the specific language you want. Can't be NULL or empty.
 * @param actualLang the actual language of the value. Can be NULL if
 * not wanted.
 * @param itemValue the localized value. Can be NULL if not wanted.
 * @param propBits the options flags describing the property. Can be NULL.
 * @return true if found, false otherwise.
 */
bool xmp_get_localized_text(XmpPtr xmp, const char *schema, const char *name,
                            const char *genericLang, const char *specificLang,
                            XmpStringPtr actualLang, XmpStringPtr itemValue,
                            uint32_t *propBits);

/** Set a localised text in a localisable property.
 * @param xmp the XMP packet
 * @param schema the schema
 * @param name the property name.
 * @param genericLang the generic language you may want to set too.
 * Can be NULL or empty.
 * @param specificLang the specific language you want. Can't be NULL or empty.
 * @param value the localized value. Cannot be NULL.
 * @param optionBits the options flags describing the property.
 * @return true if set, false otherwise.
 */
bool xmp_set_localized_text(XmpPtr xmp, const char *schema, const char *name,
                            const char *genericLang, const char *specificLang,
                            const char *value, uint32_t optionBits);

bool xmp_delete_localized_text(XmpPtr xmp, const char *schema, const char *name,
                               const char *genericLang,
                               const char *specificLang);

/** Instanciate a new string
 * @return the new instance. Must be freed with
 * xmp_string_free()
 */
XmpStringPtr xmp_string_new(void);

/** Free a XmpStringPtr
 * @param s the string to free
 */
void xmp_string_free(XmpStringPtr s);

/** Get the C string from the XmpStringPtr
 * @param s the string object
 * @return the const char * for the XmpStringPtr. It
 * belong to the object.
 */
const char * xmp_string_cstr(XmpStringPtr s);

/** Get the string length from the XmpStringPtr
 * @param s the string object
 * @return the string length. The unerlying implementation has it.
 */
size_t xmp_string_len(XmpStringPtr s);

/** Create a new iterator.
 * @param xmp the packet
 * @param schema the property schema
 * @param propName the property name
 * @param options
 * @return an iterator must be freed with % xmp_iterator_free
 */
XmpIteratorPtr xmp_iterator_new(XmpPtr xmp, const char *schema,
                                const char *propName, XmpIterOptions options);

/** Free an iterator.
 * @param iter the iterator to free.
 */
bool xmp_iterator_free(XmpIteratorPtr iter);

/** Iterate to the next value
 * @param iter the iterator
 * @param schema the schema name. Pass NULL if not wanted
 * @param propName the property path. Pass NULL if not wanted
 * @param propValue the value of the property. Pass NULL if not wanted.
 * @param options the options for the property. Pass NULL if not wanted.
 * @return true if still something, false if none
 */
bool xmp_iterator_next(XmpIteratorPtr iter, XmpStringPtr schema,
                       XmpStringPtr propName, XmpStringPtr propValue,
                       uint32_t *options);

/**
 */
bool xmp_iterator_skip(XmpIteratorPtr iter, XmpIterSkipOptions options);

/** Compare two XmpDateTime
 * @param left value
 * @param right value
 * @return if left < right, return < 0. If left > right, return > 0.
 * if left == right, return 0.
 */
int xmp_datetime_compare(XmpDateTime *left, XmpDateTime *right);

#ifdef __cplusplus
}
#endif

#endif