o Moved personal dev environment from older gcc to newer clang. Many small changes...
[asdcplib.git] / src / KLV.h
1 /*
2 Copyright (c) 2005-2011, John Hurst
3 All rights reserved.
4
5 Redistribution and use in source and binary forms, with or without
6 modification, are permitted provided that the following conditions
7 are met:
8 1. Redistributions of source code must retain the above copyright
9    notice, this list of conditions and the following disclaimer.
10 2. Redistributions in binary form must reproduce the above copyright
11    notice, this list of conditions and the following disclaimer in the
12    documentation and/or other materials provided with the distribution.
13 3. The name of the author may not be used to endorse or promote products
14    derived from this software without specific prior written permission.
15
16 THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
17 IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
18 OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
19 IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
20 INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
21 NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
22 DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
23 THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
24 (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
25 THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
26 */
27 /*! \file    KLV.h
28   \version $Id$
29   \brief   KLV objects
30 */
31
32 #ifndef _KLV_H_
33 #define _KLV_H_
34
35 #include <KM_fileio.h>
36 #include <KM_memio.h>
37 #include "AS_DCP.h"
38 #include "MDD.h"
39 #include <map>
40
41
42 namespace ASDCP
43 {
44   const ui32_t MXF_BER_LENGTH = 4;
45   const ui32_t MXF_TAG_LENGTH = 2;
46   const ui32_t SMPTE_UL_LENGTH = 16;
47   const ui32_t SMPTE_UMID_LENGTH = 32;
48   const byte_t SMPTE_UL_START[4] = { 0x06, 0x0e, 0x2b, 0x34 };
49
50 #ifndef MAX_KLV_PACKET_LENGTH
51   const ui32_t MAX_KLV_PACKET_LENGTH = 1024*1024*64;
52 #endif
53
54   const ui32_t IdentBufferLen = 128;
55   const ui32_t IntBufferLen = 64;
56
57 inline const char* i64sz(i64_t i, char* buf)
58
59   assert(buf);
60 #ifdef WIN32
61   snprintf(buf, IntBufferLen, "%I64d", i);
62 #else
63   snprintf(buf, IntBufferLen, "%lld", i);
64 #endif
65   return buf;
66 }
67
68 inline const char* ui64sz(ui64_t i, char* buf)
69
70   assert(buf);
71 #ifdef WIN32
72   snprintf(buf, IntBufferLen, "%I64u", i);
73 #else
74   snprintf(buf, IntBufferLen, "%llu", i);
75 #endif
76   return buf;
77 }
78
79   struct TagValue
80   {
81     byte_t a;
82     byte_t b;
83
84     inline bool operator<(const TagValue& rhs) const {
85       if ( a < rhs.a ) return true;
86       if ( a == rhs.a && b < rhs.b ) return true;
87       return false;
88     }
89
90     inline bool operator==(const TagValue& rhs) const {
91       if ( a != rhs.a ) return false;
92       if ( b != rhs.b ) return false;
93       return true;
94     }
95   };
96
97   using Kumu::UUID;
98
99   // Universal Label
100   class UL : public Kumu::Identifier<SMPTE_UL_LENGTH>
101     {
102     public:
103       UL() {}
104       UL(const UL& rhs) : Kumu::Identifier<SMPTE_UL_LENGTH>(rhs) {}
105       UL(const byte_t* value) : Kumu::Identifier<SMPTE_UL_LENGTH>(value) {}
106       virtual ~UL() {}
107
108       const char* EncodeString(char* str_buf, ui32_t buf_len) const;
109       bool operator==(const UL& rhs) const;
110       bool MatchIgnoreStream(const UL& rhs) const;
111       bool ExactMatch(const UL& rhs) const;
112     };
113
114   // UMID
115   class UMID : public Kumu::Identifier<SMPTE_UMID_LENGTH>
116     {
117     public:
118       UMID() {}
119       UMID(const UMID& rhs) : Kumu::Identifier<SMPTE_UMID_LENGTH>(rhs) {}
120       UMID(const byte_t* value) : Kumu::Identifier<SMPTE_UMID_LENGTH>(value) {}
121       virtual ~UMID() {}
122
123       void MakeUMID(int Type);
124       void MakeUMID(int Type, const UUID& ID);
125       const char* EncodeString(char* str_buf, ui32_t buf_len) const;
126     };
127
128   const byte_t nil_UMID[SMPTE_UMID_LENGTH] = {0};
129   const UMID NilUMID(nil_UMID);
130
131   //
132   struct MDDEntry
133   {
134     byte_t        ul[SMPTE_UL_LENGTH];
135     TagValue      tag;
136     bool          optional;
137     const char*   name;
138   };
139
140   const MDDEntry& MXFInterop_OPAtom_Entry();
141   const MDDEntry& SMPTE_390_OPAtom_Entry();
142
143   //
144   class Dictionary
145     {
146       std::map<ASDCP::UL, ui32_t>   m_md_lookup;
147       std::map<std::string, ui32_t> m_md_sym_lookup;
148       std::map<ui32_t, ASDCP::UL>   m_md_rev_lookup;
149
150       ASDCP_NO_COPY_CONSTRUCT(Dictionary);
151
152     public:
153       MDDEntry m_MDD_Table[(ui32_t)ASDCP::MDD_Max];
154
155       Dictionary();
156       ~Dictionary();
157
158       void Init();
159       bool AddEntry(const MDDEntry& Entry, ui32_t index);
160       bool DeleteEntry(ui32_t index);
161
162       const MDDEntry* FindUL(const byte_t*) const;
163       const MDDEntry* FindSymbol(const std::string&) const;
164       const MDDEntry& Type(MDD_t type_id) const;
165
166       inline const byte_t* ul(MDD_t type_id) const {
167         return Type(type_id).ul;
168       }
169
170       void Dump(FILE* = 0) const;
171     };
172
173
174   const Dictionary& DefaultSMPTEDict();
175   const Dictionary& DefaultInteropDict();
176   const Dictionary& DefaultCompositeDict();
177
178   void default_md_object_init();
179
180   //
181   class IPrimerLookup
182     {
183     public:
184       virtual ~IPrimerLookup() {}
185       virtual void     ClearTagList() = 0;
186       virtual Result_t InsertTag(const MDDEntry& Entry, ASDCP::TagValue& Tag) = 0;
187       virtual Result_t TagForKey(const ASDCP::UL& Key, ASDCP::TagValue& Tag) = 0;
188     };
189
190   //
191   class KLVPacket
192     {
193       ASDCP_NO_COPY_CONSTRUCT(KLVPacket);
194
195     protected:
196       const byte_t* m_KeyStart;
197       ui32_t        m_KLLength;
198       const byte_t* m_ValueStart;
199       ui64_t        m_ValueLength;
200       UL m_UL;
201
202     public:
203       KLVPacket() : m_KeyStart(0), m_KLLength(0), m_ValueStart(0), m_ValueLength(0) {}
204       virtual ~KLVPacket() {}
205
206       inline ui64_t  PacketLength() {
207         return m_KLLength + m_ValueLength;
208       }
209
210       inline ui64_t   ValueLength() {
211         return m_ValueLength;
212       }
213
214       inline ui32_t   KLLength() {
215         return m_KLLength;
216       }
217
218       virtual UL       GetUL();
219       virtual bool     SetUL(const UL&);
220       virtual bool     HasUL(const byte_t*);
221       virtual Result_t InitFromBuffer(const byte_t*, ui32_t);
222       virtual Result_t InitFromBuffer(const byte_t*, ui32_t, const UL& label);
223       virtual Result_t WriteKLToBuffer(ASDCP::FrameBuffer&, const UL& label, ui32_t length);
224
225       virtual Result_t WriteKLToBuffer(ASDCP::FrameBuffer& fb, ui32_t length)
226       {
227         if ( ! m_UL.HasValue() )
228           {
229             return RESULT_STATE;
230           }
231
232         return WriteKLToBuffer(fb, m_UL, length);
233       }
234
235       virtual void     Dump(FILE*, const Dictionary& Dict, bool show_value);
236     };
237
238   //
239   class KLVFilePacket : public KLVPacket
240     {
241       ASDCP_NO_COPY_CONSTRUCT(KLVFilePacket);
242
243     public:
244       ASDCP::FrameBuffer m_Buffer;
245
246       KLVFilePacket() {}
247       virtual ~KLVFilePacket() {}
248
249       virtual Result_t InitFromFile(const Kumu::FileReader&);
250       virtual Result_t InitFromFile(const Kumu::FileReader&, const UL& label);
251       virtual Result_t WriteKLToFile(Kumu::FileWriter& Writer, const UL& label, ui32_t length);
252     };
253
254 } // namespace ASDCP
255
256 #endif // _KLV_H_
257
258
259 //
260 // end KLV.h
261 //