package org.mp4parser.boxes.iso23001.part7; import org.mp4parser.support.AbstractFullBox; import org.mp4parser.tools.CastUtils; import org.mp4parser.tools.IsoTypeReader; import org.mp4parser.tools.IsoTypeWriter; import org.mp4parser.tools.UUIDConverter; import java.nio.ByteBuffer; import java.util.ArrayList; import java.util.List; import java.util.UUID; /** * <h1>4cc = "{@value #TYPE}"</h1> * <p>This box contains information needed by a Content Protection System to play back the content. The * data format is specified by the system identified by the ‘pssh’ parameter SystemID, and is considered * opaque for the purposes of this specification.</p> * <p>The data encapsulated in the Data field may be read by the identified Content Protection System to * enable decryption key acquisition and decryption of media data. For license/rights-based systems, the * header information may include data such as the URL of license server(s) or rights issuer(s) used, * embedded licenses/rights, and/or other protection system specific metadata.</p> * <p>A single file may be constructed to be playable by multiple key and digital rights management (DRM) * systems, by including one Protection System-Specific Header box for each system supported. Readers * that process such presentations must match the SystemID field in this box to the SystemID(s) of the * DRM System(s) they support, and select or create the matching Protection System-Specific Header * box(es) for storage and retrieval of Protection-Specific information interpreted or created by that DRM * system.</p> */ public class ProtectionSystemSpecificHeaderBox extends AbstractFullBox { public static final String TYPE = "pssh"; public static byte[] OMA2_SYSTEM_ID = UUIDConverter.convert(UUID.fromString("A2B55680-6F43-11E0-9A3F-0002A5D5C51B")); public static byte[] WIDEVINE = UUIDConverter.convert(UUID.fromString("edef8ba9-79d6-4ace-a3c8-27dcd51d21ed")); public static byte[] PLAYREADY_SYSTEM_ID = UUIDConverter.convert(UUID.fromString("9A04F079-9840-4286-AB92-E65BE0885F95")); byte[] content; byte[] systemId; List<UUID> keyIds = new ArrayList<UUID>(); public ProtectionSystemSpecificHeaderBox(byte[] systemId, byte[] content) { super(TYPE); this.content = content; this.systemId = systemId; } public ProtectionSystemSpecificHeaderBox() { super(TYPE); } public List<UUID> getKeyIds() { return keyIds; } public void setKeyIds(List<UUID> keyIds) { this.keyIds = keyIds; } public byte[] getSystemId() { return systemId; } public void setSystemId(byte[] systemId) { assert systemId.length == 16; this.systemId = systemId; } public byte[] getContent() { return content; } public void setContent(byte[] content) { this.content = content; } @Override protected long getContentSize() { long l = 24 + content.length; if (getVersion() > 0) { l += 4; l += 16 * keyIds.size(); } return l; } @Override protected void getContent(ByteBuffer byteBuffer) { writeVersionAndFlags(byteBuffer); assert systemId.length == 16; byteBuffer.put(systemId, 0, 16); if (getVersion() > 0) { IsoTypeWriter.writeUInt32(byteBuffer, keyIds.size()); for (UUID keyId : keyIds) { byteBuffer.put(UUIDConverter.convert(keyId)); } } IsoTypeWriter.writeUInt32(byteBuffer, content.length); byteBuffer.put(content); } @Override protected void _parseDetails(ByteBuffer content) { parseVersionAndFlags(content); systemId = new byte[16]; content.get(systemId); if (getVersion() > 0) { int count = CastUtils.l2i(IsoTypeReader.readUInt32(content)); while (count-- > 0) { byte[] k = new byte[16]; content.get(k); keyIds.add(UUIDConverter.convert(k)); } } long length = IsoTypeReader.readUInt32(content); this.content = new byte[content.remaining()]; content.get(this.content); assert length == this.content.length; } }