I pronounce it Good!

git-svn-id: file:///srv/svn/repos/haiku/trunk/current@6275 a95241bf-73f2-0310-859d-f6bbb57e9c96
This commit is contained in:
shatty 2004-01-25 11:37:10 +00:00
parent fbe89ed67b
commit 170d6e3fbc
5 changed files with 113 additions and 76 deletions

View File

@ -1,9 +1,10 @@
#include "OggFrameInfo.h"
OggFrameInfo::OggFrameInfo(uint page, uint packet)
OggFrameInfo::OggFrameInfo(uint page, uint packetonpage, uint packet)
{
this->page = page;
this->packetonpage = packetonpage;
this->packet = packet;
}
@ -19,6 +20,12 @@ OggFrameInfo::GetPage() const
return page;
}
uint
OggFrameInfo::GetPacketOnPage() const
{
return packetonpage;
}
uint
OggFrameInfo::GetPacket() const
{

View File

@ -5,14 +5,16 @@
class OggFrameInfo {
public:
OggFrameInfo(uint page, uint packet);
OggFrameInfo(uint page, uint packetonpage, uint packet);
virtual ~OggFrameInfo();
uint GetPage() const;
uint GetPacketOnPage() const;
uint GetPacket() const;
private:
uint page; // the page the packet started on
uint packet; // of all the packets on that page which packet is this
uint page; // the page the frame started on
uint packetonpage; // of all the packets on that page which packet is this
uint packet; // the packet the frame started on
};
#endif _OGG_FRAME_INFO_H

View File

@ -156,7 +156,6 @@ OggReader::GetPageAt(off_t position, ogg_stream_state * stream, int read_size)
while ((result = ogg_sync_pageout(&sync,&page)) == 0) {
char * buffer = ogg_sync_buffer(&sync,read_size);
ssize_t bytes = fSeekable->ReadAt(position,buffer,read_size);
position += read_size;
if (bytes == 0) {
TRACE("OggReader::GetPage: Read: no data\n");
return B_LAST_BUFFER_ERROR;
@ -165,6 +164,7 @@ OggReader::GetPageAt(off_t position, ogg_stream_state * stream, int read_size)
TRACE("OggReader::GetPage: Read: error\n");
return bytes;
}
position += bytes;
if (ogg_sync_wrote(&sync,bytes) != 0) {
TRACE("OggReader::GetPage: ogg_sync_wrote failed?: error\n");
return B_ERROR;

View File

@ -3,6 +3,7 @@
#include "OggTheoraStream.h"
#include "OggTobiasStream.h"
#include "OggVorbisStream.h"
#include <Autolock.h>
#include <stdio.h>
#define TRACE_THIS 1
@ -56,20 +57,33 @@ OggStream::findIdentifier(const ogg_packet & packet, const char * id, uint pos)
OggStream::OggStream(long serialno)
{
TRACE("OggStream::OggStream\n");
this->fSerialno = serialno;
fCurrentFrame = 0;
fCurrentTime = 0;
ogg_stream_init(&fStreamState,serialno);
fCurrentPage = 0;
fCurrentPacket = 0;
this->fSerialno = serialno;
ogg_sync_init(&fSync);
ogg_stream_init(&fSeekStreamState,serialno);
fCurrentPage = 0;
fPacketOnCurrentPage = 0;
fCurrentPacket = 0;
ogg_stream_init(&fEndStreamState,serialno);
fEndPage = 0;
fPacketOnEndPage = 0;
fEndPacket = 0;
}
OggStream::~OggStream()
{
ogg_stream_clear(&fStreamState);
// free internal header packet storage
std::vector<ogg_packet>::iterator iter = fHeaderPackets.begin();
while (iter != fHeaderPackets.end()) {
delete iter->packet;
iter++;
}
// free internal stream state storage
ogg_sync_clear(&fSync);
ogg_stream_clear(&fSeekStreamState);
ogg_stream_clear(&fEndStreamState);
}
@ -87,7 +101,14 @@ OggStream::AddPage(off_t position, ogg_page * page)
if (position >= 0) {
fPagePositions.push_back(position);
}
ogg_stream_pagein(&fStreamState,page);
BAutolock autolock(fSyncLock);
char * buffer;
buffer = ogg_sync_buffer(&fSync,page->header_len);
memcpy(buffer,page->header,page->header_len);
ogg_sync_wrote(&fSync,page->header_len);
buffer = ogg_sync_buffer(&fSync,page->body_len);
memcpy(buffer,page->body,page->body_len);
ogg_sync_wrote(&fSync,page->body_len);
return B_OK;
}
@ -135,7 +156,7 @@ OggStream::GetStreamInfo(int64 *frameCount, bigtime_t *duration,
strncpy((char*)format->user_data, (char*)(&packet), 4);
format->SetMetaData((void*)&fHeaderPackets,sizeof(fHeaderPackets));
*duration = 80000000;
*duration = 140000000;
*frameCount = 60000;
return B_OK;
}
@ -146,8 +167,14 @@ OggStream::Seek(uint32 seekTo, int64 *frame, bigtime_t *time)
{
TRACE("OggStream::Seek to %lld : %lld\n", *frame, *time);
if (seekTo & B_MEDIA_SEEK_TO_FRAME) {
if (*frame > fOggFrameInfos.size() - 1) {
// seek to end
fCurrentPage = fEndPage;
fPacketOnCurrentPage = fPacketOnEndPage;
fCurrentPacket = fEndPacket;
return B_OK;
}
*frame = max_c(0, *frame); // clip to zero
*frame = min_c(*frame, fOggFrameInfos.size()-1); // clip to max
// input the page into a temporary seek stream
ogg_stream_state seekStreamState;
uint pageno = fOggFrameInfos[*frame].GetPage();
@ -159,13 +186,11 @@ OggStream::Seek(uint32 seekTo, int64 *frame, bigtime_t *time)
return result; // pageno/fPagePosition corrupted?
}
// discard earlier packets from this page
uint packetno = fOggFrameInfos[*frame].GetPacket();
uint packetno = fOggFrameInfos[*frame].GetPacketOnPage();
while (packetno-- > 0) {
ogg_packet packet;
if (ogg_stream_packetout(&fSeekStreamState, &packet) != 1) {
TRACE("OggStream::GetPageAt packetno corrupt?\n");
return B_ERROR; // packetno corrupted?
}
// don't check for errors, we may have a packet ending on page
ogg_stream_packetout(&seekStreamState, &packet);
}
// clear out the former seek stream state
// this will delete its internal storage
@ -175,7 +200,9 @@ OggStream::Seek(uint32 seekTo, int64 *frame, bigtime_t *time)
fSeekStreamState = seekStreamState;
// we notably do not clear our temporary stream
// instead we just let it go out of scope
fCurrentFrame = *frame;
fCurrentPage = fOggFrameInfos[*frame].GetPage();
fPacketOnCurrentPage = fOggFrameInfos[*frame].GetPacketOnPage();
fCurrentPacket = fOggFrameInfos[*frame].GetPacket();
} else if (seekTo & B_MEDIA_SEEK_TO_TIME) {
*frame = *time/50000;
return Seek(B_MEDIA_SEEK_TO_FRAME,frame,time);
@ -189,9 +216,13 @@ OggStream::GetNextChunk(void **chunkBuffer, int32 *chunkSize,
media_header *mediaHeader)
{
static ogg_packet packet;
if (fCurrentPacket - fHeaderPackets.size() == fOggFrameInfos.size()) {
OggFrameInfo info(fEndPage,fPacketOnEndPage,fEndPacket);
fOggFrameInfos.push_back(info);
}
status_t result = GetPacket(&packet);
if (result != B_OK) {
TRACE("OggStream::GetNextChunk failed: GetPacket failed\n");
TRACE("OggStream::GetNextChunk failed: GetPacket = %s\n", strerror(result));
return result;
}
*chunkBuffer = &packet;
@ -204,42 +235,65 @@ OggStream::GetNextChunk(void **chunkBuffer, int32 *chunkSize,
status_t
OggStream::GetPacket(ogg_packet * packet)
{
if (fCurrentFrame == fOggFrameInfos.size()) {
if (fCurrentPacket >= fEndPacket) {
// at the end, pull the packet
uint old_page = fCurrentPage;
uint old_packet = fCurrentPacket;
while (ogg_stream_packetpeek(&fStreamState, NULL) != 1) {
status_t result = fReaderInterface->GetNextPage();
if (result != B_OK) {
return result;
uint8 pageno = fEndPage;
while (ogg_stream_packetpeek(&fEndStreamState, NULL) != 1) {
BAutolock autolock(fSyncLock);
int result;
ogg_page page;
while ((result = ogg_sync_pageout(&fSync,&page)) == 0) {
status_t result = fReaderInterface->GetNextPage();
if (result != B_OK) {
TRACE("OggStream::GetPacket: GetNextPage = %s\n", strerror(result));
return result;
}
}
fCurrentPage++;
if (result == -1) {
TRACE("OggStream::GetPacket: ogg_sync_pageout: not synced??\n");
return B_ERROR;
}
if (ogg_stream_pagein(&fEndStreamState,&page) != 0) {
TRACE("OggStream::GetPacket: ogg_stream_pagein: failed??\n");
return B_ERROR;
}
fEndPage++;
}
if (ogg_stream_packetout(&fStreamState, packet) != 1) {
if (ogg_stream_packetout(&fEndStreamState, packet) != 1) {
TRACE("OggStream::GetPacket: ogg_stream_packetout failed at the end\n");
return B_ERROR;
}
OggFrameInfo info(old_page, old_packet);
fOggFrameInfos.push_back(info);
if (fCurrentPage != old_page) {
fCurrentPacket = 0;
fEndPacket++;
if (pageno != fEndPage) {
fPacketOnEndPage = 0;
} else {
fCurrentPacket++;
fPacketOnEndPage++;
}
fCurrentPacket = fEndPacket;
fPacketOnCurrentPage = fPacketOnEndPage;
fCurrentPage = fEndPage;
} else {
// in the middle, get packet at position
uint pageno = fOggFrameInfos[fCurrentFrame].GetPage();
uint8 page = fCurrentPage;
while (ogg_stream_packetpeek(&fSeekStreamState, NULL) != 1) {
off_t position = fPagePositions[pageno++];
off_t position = fPagePositions[fCurrentPage++];
status_t result = fReaderInterface->GetPageAt(position, &fSeekStreamState);
if (result != B_OK) {
TRACE("OggStream::GetPacket: GetPageAt = %s\n", strerror(result));
return result;
}
}
if (ogg_stream_packetout(&fSeekStreamState, packet) != 1) {
TRACE("OggStream::GetPacket: ogg_stream_packetout failed in the middle\n");
return B_ERROR;
}
fCurrentPacket++;
if (page != fCurrentPage) {
fPacketOnCurrentPage = 0;
} else {
fPacketOnCurrentPage++;
}
}
fCurrentFrame++; // ever moving forward!
return B_OK;
}
@ -254,32 +308,3 @@ OggStream::SaveHeaderPacket(ogg_packet packet)
fHeaderPackets.push_back(packet);
}
// estimate 1 frame = 256 bytes
int64
OggStream::PositionToFrame(off_t position)
{
return position/256;
}
off_t
OggStream::FrameToPosition(int64 frame)
{
return frame*256;
}
// estimate 1 byte = 125 microseconds
bigtime_t
OggStream::PositionToTime(off_t position)
{
return position*125;
}
off_t
OggStream::TimeToPosition(bigtime_t time)
{
return time/125;
}

View File

@ -42,21 +42,24 @@ protected:
void SaveHeaderPacket(ogg_packet packet);
std::vector<ogg_packet> fHeaderPackets;
// Seek helpers
virtual int64 PositionToFrame(off_t position);
virtual off_t FrameToPosition(int64 frame);
virtual bigtime_t PositionToTime(off_t position);
virtual off_t TimeToPosition(bigtime_t time);
private:
long fSerialno;
protected:
int64 fCurrentFrame;
bigtime_t fCurrentTime;
private:
long fSerialno;
std::vector<off_t> fPagePositions;
std::vector<OggFrameInfo> fOggFrameInfos;
ogg_stream_state fStreamState;
uint fCurrentPage;
uint fCurrentPacket;
ogg_sync_state fSync;
BLocker fSyncLock;
ogg_stream_state fSeekStreamState;
uint fCurrentPage;
uint fPacketOnCurrentPage;
uint fCurrentPacket;
ogg_stream_state fEndStreamState;
uint fEndPage;
uint fPacketOnEndPage;
uint fEndPacket;
OggReader::GetPageInterface * fReaderInterface;
};