SND\AstrorEnales_cp 0d25ca7589 - Added OggUtils Support project by making use of the csogg and csvorbis projects
- Implemented native Song playback in OpenAL and XAudio2
- Some tweaks in the MediaPlayer and MediaQueue
- Added a testmusic.ogg file to the media folder
2015-03-15 01:11:27 +01:00

309 lines
6.8 KiB
C#

/* csvorbis
* Copyright (C) 2000 ymnk, JCraft,Inc.
*
* Written by: 2000 ymnk<ymnk@jcraft.com>
* Ported to C# from JOrbis by: Mark Crichton <crichton@gimp.org>
*
* Thanks go to the JOrbis team, for licencing the code under the
* LGPL, making my job a lot easier.
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU Library General Public License
* as published by the Free Software Foundation; either version 2 of
* the License, or (at your option) any later version.
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Library General Public License for more details.
*
* You should have received a copy of the GNU Library General Public
* License along with this program; if not, write to the Free Software
* Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
*/
using System;
using System.Text;
using csogg;
namespace csvorbis
{
// the comments are not part of vorbis_info so that vorbis_info can be
// static storage
public class Comment
{
private static String _vorbis="vorbis";
//private static int OV_EFAULT=-129;
private static int OV_EIMPL=-130;
// unlimited user comment fields. libvorbis writes 'libvorbis'
// whatever vendor is set to in encode
public byte[][] user_comments;
public int[] comment_lengths;
public int comments;
public byte[] vendor;
public void init()
{
user_comments=null;
comments=0;
vendor=null;
}
public void add(String comment)
{
Encoding AE = Encoding.UTF8;
byte[] comment_byt = AE.GetBytes(comment);
add(comment_byt);
}
private void add(byte[] comment)
{
byte[][] foo=new byte[comments+2][];
if(user_comments!=null)
{
Array.Copy(user_comments, 0, foo, 0, comments);
}
user_comments=foo;
int[] goo=new int[comments+2];
if(comment_lengths!=null)
{
Array.Copy(comment_lengths, 0, goo, 0, comments);
}
comment_lengths=goo;
byte[] bar=new byte[comment.Length+1];
Array.Copy(comment, 0, bar, 0, comment.Length);
user_comments[comments]=bar;
comment_lengths[comments]=comment.Length;
comments++;
user_comments[comments]=null;
}
public void add_tag(String tag, String contents)
{
if(contents==null) contents="";
add(tag+"="+contents);
}
/*
private void add_tag(byte[] tag, byte[] contents){
byte[] foo=new byte[tag.length+contents.length+1];
int j=0;
for(int i=0; i<tag.length; i++){foo[j++]=tag[i];}
foo[j++]=(byte)'='; j++;
for(int i=0; i<contents.length; i++){foo[j++]=tag[i];}
add(foo);
}
*/
// This is more or less the same as strncasecmp - but that doesn't exist
// * everywhere, and this is a fairly trivial function, so we include it
static bool tagcompare(byte[] s1, byte[] s2, int n)
{
int c=0;
byte u1, u2;
while(c < n)
{
u1=s1[c]; u2=s2[c];
if(u1>='A')u1=(byte)(u1-'A'+'a');
if(u2>='A')u2=(byte)(u2-'A'+'a');
if(u1!=u2){ return false; }
c++;
}
return true;
}
public String query(String tag)
{
return query(tag, 0);
}
public String query(String tag, int count)
{
Encoding AE = Encoding.UTF8;
byte[] tag_byt = AE.GetBytes(tag);
int foo=query(tag_byt, count);
if(foo==-1)return null;
byte[] comment=user_comments[foo];
for(int i=0; i<comment_lengths[foo]; i++)
{
if(comment[i]=='=')
{
char[] comment_uni = AE.GetChars(comment);
return new String(comment_uni, i+1, comment_lengths[foo]-(i+1));
}
}
return null;
}
private int query(byte[] tag, int count)
{
int i=0;
int found = 0;
int taglen = tag.Length;
byte[] fulltag = new byte[taglen+2];
Array.Copy(tag, 0, fulltag, 0, tag.Length);
fulltag[tag.Length]=(byte)'=';
for(i=0;i<comments;i++)
{
if(tagcompare(user_comments[i], fulltag, taglen))
{
if(count==found)
{
// We return a pointer to the data, not a copy
//return user_comments[i] + taglen + 1;
return i;
}
else{ found++; }
}
}
return -1;
}
internal int unpack(csBuffer opb)
{
int vendorlen=opb.read(32);
if(vendorlen<0)
{
//goto err_out;
clear();
return(-1);
}
vendor=new byte[vendorlen+1];
opb.read(vendor,vendorlen);
comments=opb.read(32);
if(comments<0)
{
//goto err_out;
clear();
return(-1);
}
user_comments=new byte[comments+1][];
comment_lengths=new int[comments+1];
for(int i=0;i<comments;i++)
{
int len=opb.read(32);
if(len<0)
{
//goto err_out;
clear();
return(-1);
}
comment_lengths[i]=len;
user_comments[i]=new byte[len+1];
opb.read(user_comments[i], len);
}
if(opb.read(1)!=1)
{
//goto err_out; // EOP check
clear();
return(-1);
}
return(0);
// err_out:
// comment_clear(vc);
// return(-1);
}
int pack(csBuffer opb)
{
String temp="Xiphophorus libVorbis I 20000508";
Encoding AE = Encoding.UTF8;
byte[] temp_byt = AE.GetBytes(temp);
byte[] _vorbis_byt = AE.GetBytes(_vorbis);
// preamble
opb.write(0x03,8);
opb.write(_vorbis_byt);
// vendor
opb.write(temp.Length,32);
opb.write(temp_byt);
// comments
opb.write(comments,32);
if(comments!=0)
{
for(int i=0;i<comments;i++)
{
if(user_comments[i]!=null)
{
opb.write(comment_lengths[i],32);
opb.write(user_comments[i]);
}
else
{
opb.write(0,32);
}
}
}
opb.write(1,1);
return(0);
}
public int header_out(Packet op)
{
csBuffer opb=new csBuffer();
opb.writeinit();
if(pack(opb)!=0) return OV_EIMPL;
op.packet_base = new byte[opb.bytes()];
op.packet=0;
op.bytes=opb.bytes();
Array.Copy(opb.buf(), 0, op.packet_base, 0, op.bytes);
op.b_o_s=0;
op.e_o_s=0;
op.granulepos=0;
return 0;
}
internal void clear()
{
for(int i=0;i<comments;i++)
user_comments[i]=null;
user_comments=null;
vendor=null;
}
public String getVendor()
{
Encoding AE = Encoding.UTF8;
char[] vendor_uni = AE.GetChars(vendor);
return new String(vendor_uni);
}
public String getComment(int i)
{
Encoding AE = Encoding.UTF8;
if(comments<=i)return null;
char[] user_comments_uni = AE.GetChars(user_comments[i]);
return new String(user_comments_uni);
}
public String toString()
{
Encoding AE = Encoding.UTF8;
String long_string = "Vendor: " + new String(AE.GetChars(vendor));
for(int i=0; i < comments; i++)
long_string = long_string + "\nComment: " + new String(AE.GetChars(user_comments[i]));
long_string = long_string + "\n";
return long_string;
}
}
}