mirror of
git://xwords.git.sourceforge.net/gitroot/xwords/xwords
synced 2025-01-22 07:28:16 +01:00
move hashing into memstream impl, and reintroduce the old/broken
treatment of the final 8 bits. I'll need to release a version that still produces the incorrect hash for compatibility with existing clients that expect it, but that also looks for the correct hash. When that's out there I can do a new release that sends the correct hash.
This commit is contained in:
parent
4813c95976
commit
ebcae1a523
9 changed files with 85 additions and 128 deletions
|
@ -25,6 +25,7 @@
|
||||||
#include "comtypes.h"
|
#include "comtypes.h"
|
||||||
#include "memstream.h"
|
#include "memstream.h"
|
||||||
#include "vtabmgr.h"
|
#include "vtabmgr.h"
|
||||||
|
#include "strutils.h"
|
||||||
|
|
||||||
#ifdef CPLUS
|
#ifdef CPLUS
|
||||||
extern "C" {
|
extern "C" {
|
||||||
|
@ -184,7 +185,7 @@ mem_stream_getBits( XWStreamCtxt* p_sctx, XP_U16 nBits )
|
||||||
return result;
|
return result;
|
||||||
} /* stream_getBits */
|
} /* stream_getBits */
|
||||||
|
|
||||||
#if defined HASH_STREAM || defined DEBUG
|
#if defined DEBUG
|
||||||
static void
|
static void
|
||||||
mem_stream_copyBits( const XWStreamCtxt* p_sctx, XWStreamPos endPos,
|
mem_stream_copyBits( const XWStreamCtxt* p_sctx, XWStreamPos endPos,
|
||||||
XP_U8* buf, XP_U16* lenp )
|
XP_U8* buf, XP_U16* lenp )
|
||||||
|
@ -371,6 +372,36 @@ mem_stream_getSize( const XWStreamCtxt* p_sctx )
|
||||||
return size;
|
return size;
|
||||||
} /* mem_stream_getSize */
|
} /* mem_stream_getSize */
|
||||||
|
|
||||||
|
static XP_U32
|
||||||
|
mem_stream_getHash( const XWStreamCtxt* p_sctx, XWStreamPos pos,
|
||||||
|
XP_Bool correct )
|
||||||
|
{
|
||||||
|
XP_U32 hash = 0;
|
||||||
|
const MemStreamCtxt* stream = (const MemStreamCtxt*)p_sctx;
|
||||||
|
const XP_U8* ptr = stream->buf;
|
||||||
|
XP_U16 len = BYTE_PART(pos);
|
||||||
|
XP_U16 bits = BIT_PART(pos);
|
||||||
|
if ( 0 != bits ) {
|
||||||
|
XP_ASSERT( 0 < len );
|
||||||
|
--len;
|
||||||
|
}
|
||||||
|
|
||||||
|
hash = augmentHash( 0, ptr, len );
|
||||||
|
if ( 0 != bits ) {
|
||||||
|
XP_U8 byt = ptr[len];
|
||||||
|
if ( correct ) {
|
||||||
|
byt &= ~(0xFF << bits);
|
||||||
|
} else {
|
||||||
|
byt &= 1 << bits;
|
||||||
|
}
|
||||||
|
hash = augmentHash( hash, &byt, 1 );
|
||||||
|
}
|
||||||
|
hash = finishHash( hash );
|
||||||
|
|
||||||
|
LOG_RETURNF( "%X(%d:%d)", hash, len, bits );
|
||||||
|
return hash;
|
||||||
|
} /* mem_stream_getHash */
|
||||||
|
|
||||||
static const XP_U8*
|
static const XP_U8*
|
||||||
mem_stream_getPtr( const XWStreamCtxt* p_sctx )
|
mem_stream_getPtr( const XWStreamCtxt* p_sctx )
|
||||||
{
|
{
|
||||||
|
@ -474,7 +505,7 @@ make_vtable( MemStreamCtxt* stream )
|
||||||
SET_VTABLE_ENTRY( vtable, stream_getU16, mem );
|
SET_VTABLE_ENTRY( vtable, stream_getU16, mem );
|
||||||
SET_VTABLE_ENTRY( vtable, stream_getU32, mem );
|
SET_VTABLE_ENTRY( vtable, stream_getU32, mem );
|
||||||
SET_VTABLE_ENTRY( vtable, stream_getBits, mem );
|
SET_VTABLE_ENTRY( vtable, stream_getBits, mem );
|
||||||
#if defined HASH_STREAM || defined DEBUG
|
#if defined DEBUG
|
||||||
SET_VTABLE_ENTRY( vtable, stream_copyBits, mem );
|
SET_VTABLE_ENTRY( vtable, stream_copyBits, mem );
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
@ -495,6 +526,7 @@ make_vtable( MemStreamCtxt* stream )
|
||||||
SET_VTABLE_ENTRY( vtable, stream_close, mem );
|
SET_VTABLE_ENTRY( vtable, stream_close, mem );
|
||||||
|
|
||||||
SET_VTABLE_ENTRY( vtable, stream_getSize, mem );
|
SET_VTABLE_ENTRY( vtable, stream_getSize, mem );
|
||||||
|
SET_VTABLE_ENTRY( vtable, stream_getHash, mem );
|
||||||
SET_VTABLE_ENTRY( vtable, stream_getPtr, mem );
|
SET_VTABLE_ENTRY( vtable, stream_getPtr, mem );
|
||||||
SET_VTABLE_ENTRY( vtable, stream_getAddress, mem );
|
SET_VTABLE_ENTRY( vtable, stream_getAddress, mem );
|
||||||
SET_VTABLE_ENTRY( vtable, stream_setAddress, mem );
|
SET_VTABLE_ENTRY( vtable, stream_setAddress, mem );
|
||||||
|
|
|
@ -312,33 +312,22 @@ model_destroy( ModelCtxt* model )
|
||||||
} /* model_destroy */
|
} /* model_destroy */
|
||||||
|
|
||||||
XP_U32
|
XP_U32
|
||||||
model_getHash( const ModelCtxt* model, XP_U16 version )
|
model_getHash( const ModelCtxt* model )
|
||||||
{
|
{
|
||||||
#ifndef STREAM_VERS_HASHSTREAM
|
#ifndef STREAM_VERS_HASHSTREAM
|
||||||
XP_USE(version);
|
XP_USE(version);
|
||||||
#endif
|
#endif
|
||||||
StackCtxt* stack = model->vol.stack;
|
StackCtxt* stack = model->vol.stack;
|
||||||
XP_ASSERT( !!stack );
|
XP_ASSERT( !!stack );
|
||||||
XP_U32 hash =
|
return stack_getHash( stack, XP_FALSE );
|
||||||
#ifdef STREAM_VERS_HASHSTREAM
|
|
||||||
STREAM_VERS_HASHSTREAM <= version ?
|
|
||||||
stack_getHash( stack ) :
|
|
||||||
#endif
|
|
||||||
stack_getHashOld( stack );
|
|
||||||
/* XP_LOGF( "%s(version=%x)=>%.8X", __func__, version, */
|
|
||||||
/* (unsigned int)hash ); */
|
|
||||||
return hash;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
XP_Bool
|
XP_Bool
|
||||||
model_hashMatches( const ModelCtxt* model, const XP_U32 hash )
|
model_hashMatches( const ModelCtxt* model, const XP_U32 hash )
|
||||||
{
|
{
|
||||||
StackCtxt* stack = model->vol.stack;
|
StackCtxt* stack = model->vol.stack;
|
||||||
XP_Bool matches =
|
XP_Bool matches = hash == stack_getHash( stack, XP_TRUE )
|
||||||
#ifdef STREAM_VERS_HASHSTREAM
|
|| hash == stack_getHash( stack, XP_FALSE );
|
||||||
(hash == stack_getHash( stack )) ||
|
|
||||||
#endif
|
|
||||||
(hash == stack_getHashOld( stack ));
|
|
||||||
return matches;
|
return matches;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -349,18 +338,11 @@ model_popToHash( ModelCtxt* model, const XP_U32 hash, PoolContext* pool )
|
||||||
StackCtxt* stack = model->vol.stack;
|
StackCtxt* stack = model->vol.stack;
|
||||||
const XP_U16 nEntries = stack_getNEntries( stack );
|
const XP_U16 nEntries = stack_getNEntries( stack );
|
||||||
StackEntry entries[nEntries];
|
StackEntry entries[nEntries];
|
||||||
#ifdef DEBUG
|
|
||||||
XP_U32 hashes[nEntries];
|
|
||||||
#endif
|
|
||||||
XP_S16 foundAt = -1;
|
XP_S16 foundAt = -1;
|
||||||
|
|
||||||
for ( XP_U16 ii = 0; ii < nEntries; ++ii ) {
|
for ( XP_U16 ii = 0; ii < nEntries; ++ii ) {
|
||||||
XP_U32 thisHash =
|
if ( hash == stack_getHash( stack, XP_TRUE )
|
||||||
#ifdef DEBUG
|
|| hash == stack_getHash( stack, XP_FALSE ) ) {
|
||||||
hashes[ii] =
|
|
||||||
#endif
|
|
||||||
stack_getHash( stack );
|
|
||||||
if ( hash == thisHash ) {
|
|
||||||
foundAt = ii;
|
foundAt = ii;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
@ -372,8 +354,6 @@ model_popToHash( ModelCtxt* model, const XP_U32 hash, PoolContext* pool )
|
||||||
|
|
||||||
for ( XP_S16 ii = nPopped - 1; ii >= 0; --ii ) {
|
for ( XP_S16 ii = nPopped - 1; ii >= 0; --ii ) {
|
||||||
stack_redo( stack, &entries[ii] );
|
stack_redo( stack, &entries[ii] );
|
||||||
/* Assert not needed for long */
|
|
||||||
XP_ASSERT( hashes[ii] = stack_getHash( stack ) );
|
|
||||||
}
|
}
|
||||||
|
|
||||||
XP_Bool found = -1 != foundAt;
|
XP_Bool found = -1 != foundAt;
|
||||||
|
@ -386,7 +366,8 @@ model_popToHash( ModelCtxt* model, const XP_U32 hash, PoolContext* pool )
|
||||||
model_undoLatestMoves( model, pool, foundAt, NULL, NULL );
|
model_undoLatestMoves( model, pool, foundAt, NULL, NULL );
|
||||||
XP_ASSERT( success );
|
XP_ASSERT( success );
|
||||||
/* Assert not needed for long */
|
/* Assert not needed for long */
|
||||||
XP_ASSERT( hash == stack_getHash( model->vol.stack ) );
|
XP_ASSERT( hash == stack_getHash( model->vol.stack, XP_TRUE )
|
||||||
|
|| hash == stack_getHash( model->vol.stack, XP_FALSE ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
return found;
|
return found;
|
||||||
|
|
|
@ -122,7 +122,7 @@ void model_writeToTextStream( const ModelCtxt* model, XWStreamCtxt* stream );
|
||||||
|
|
||||||
void model_setSize( ModelCtxt* model, XP_U16 boardSize );
|
void model_setSize( ModelCtxt* model, XP_U16 boardSize );
|
||||||
void model_destroy( ModelCtxt* model );
|
void model_destroy( ModelCtxt* model );
|
||||||
XP_U32 model_getHash( const ModelCtxt* model, XP_U16 version );
|
XP_U32 model_getHash( const ModelCtxt* model );
|
||||||
XP_Bool model_hashMatches( const ModelCtxt* model, XP_U32 hash );
|
XP_Bool model_hashMatches( const ModelCtxt* model, XP_U32 hash );
|
||||||
XP_Bool model_popToHash( ModelCtxt* model, const XP_U32 hash,
|
XP_Bool model_popToHash( ModelCtxt* model, const XP_U32 hash,
|
||||||
PoolContext* pool );
|
PoolContext* pool );
|
||||||
|
|
|
@ -67,101 +67,13 @@ stack_init( StackCtxt* stack )
|
||||||
shrunk to fit as soon as we serialize/deserialize anyway. */
|
shrunk to fit as soon as we serialize/deserialize anyway. */
|
||||||
} /* stack_init */
|
} /* stack_init */
|
||||||
|
|
||||||
static XP_U32
|
|
||||||
augmentHash( XP_U32 hash, const XP_U8* ptr, XP_U16 len )
|
|
||||||
{
|
|
||||||
XP_ASSERT( 0 < len );
|
|
||||||
// see http://en.wikipedia.org/wiki/Jenkins_hash_function
|
|
||||||
XP_U16 ii;
|
|
||||||
for ( ii = 0; ii < len; ++ii ) {
|
|
||||||
hash += *ptr++;
|
|
||||||
hash += (hash << 10);
|
|
||||||
hash ^= (hash >> 6);
|
|
||||||
}
|
|
||||||
#ifdef DEBUG_HASHING
|
|
||||||
XP_LOGF( "%s: hashed %d bytes -> %X", __func__, len, (unsigned int)hash );
|
|
||||||
#endif
|
|
||||||
return hash;
|
|
||||||
}
|
|
||||||
|
|
||||||
static XP_U32
|
|
||||||
finishHash( XP_U32 hash )
|
|
||||||
{
|
|
||||||
hash += (hash << 3);
|
|
||||||
hash ^= (hash >> 11);
|
|
||||||
hash += (hash << 15);
|
|
||||||
return hash;
|
|
||||||
}
|
|
||||||
|
|
||||||
static XP_U32
|
|
||||||
augmentFor( XP_U32 hash, const StackEntry* entry )
|
|
||||||
{
|
|
||||||
switch( entry->moveType ) {
|
|
||||||
case ASSIGN_TYPE: {
|
|
||||||
TrayTileSet tiles;
|
|
||||||
sortTiles( &tiles, &entry->u.assign.tiles, 0 );
|
|
||||||
hash = augmentHash( hash, (XP_U8*)&tiles, sizeof(tiles) );
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case MOVE_TYPE:
|
|
||||||
hash = augmentHash( hash, (XP_U8*)&entry->u.move,
|
|
||||||
sizeof(entry->u.move) );
|
|
||||||
break;
|
|
||||||
case TRADE_TYPE:
|
|
||||||
hash = augmentHash( hash, (XP_U8*)&entry->u.trade,
|
|
||||||
sizeof(entry->u.trade) );
|
|
||||||
break;
|
|
||||||
case PHONY_TYPE:
|
|
||||||
hash = augmentHash( hash, (XP_U8*)&entry->u.phony,
|
|
||||||
sizeof(entry->u.phony) );
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
return hash;
|
|
||||||
}
|
|
||||||
|
|
||||||
XP_U32
|
|
||||||
stack_getHashOld( StackCtxt* stack )
|
|
||||||
{
|
|
||||||
XP_U16 nn, nEntries = stack->nEntries;
|
|
||||||
XP_U32 hash = 0L;
|
|
||||||
for ( nn = 0; nn < nEntries; ++nn ) {
|
|
||||||
StackEntry entry;
|
|
||||||
XP_MEMSET( &entry, 0, sizeof(entry) );
|
|
||||||
if ( !stack_getNthEntry( stack, nn, &entry ) ) {
|
|
||||||
XP_ASSERT( 0 );
|
|
||||||
}
|
|
||||||
hash = augmentFor( hash, &entry );
|
|
||||||
#ifdef DEBUG_HASHING
|
|
||||||
XP_LOGF( "%s: hash after %d: %.8X", __func__, nn, (unsigned int)hash );
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
XP_ASSERT( 0 != hash );
|
|
||||||
hash = finishHash( hash );
|
|
||||||
#ifdef DEBUG_HASHING
|
|
||||||
LOG_RETURNF( "%.8X", (unsigned int)hash );
|
|
||||||
#endif
|
|
||||||
return hash;
|
|
||||||
} /* stack_getHashOld */
|
|
||||||
|
|
||||||
#ifdef STREAM_VERS_HASHSTREAM
|
#ifdef STREAM_VERS_HASHSTREAM
|
||||||
XP_U32
|
XP_U32
|
||||||
stack_getHash( const StackCtxt* stack )
|
stack_getHash( const StackCtxt* stack, XP_Bool correct )
|
||||||
{
|
{
|
||||||
XP_U32 hash = 0;
|
XP_U32 hash = 0;
|
||||||
if ( !!stack->data ) {
|
if ( !!stack->data ) {
|
||||||
XP_U16 len = 0;
|
hash = stream_getHash( stack->data, stack->top, correct );
|
||||||
stream_copyBits( stack->data, stack->top, NULL, &len );
|
|
||||||
if ( 0 < len ) {
|
|
||||||
XP_U8 buf[len];
|
|
||||||
stream_copyBits( stack->data, stack->top, buf, &len );
|
|
||||||
#ifdef DEBUG_HASHING
|
|
||||||
LOG_HEX( buf, len, __func__ );
|
|
||||||
#endif
|
|
||||||
hash = finishHash( augmentHash( 0L, buf, len ) );
|
|
||||||
#ifdef DEBUG_HASHING
|
|
||||||
LOG_RETURNF( "%.8X", (unsigned int)hash );
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
return hash;
|
return hash;
|
||||||
} /* stack_getHash */
|
} /* stack_getHash */
|
||||||
|
@ -332,7 +244,7 @@ static void
|
||||||
pushEntry( StackCtxt* stack, const StackEntry* entry )
|
pushEntry( StackCtxt* stack, const StackEntry* entry )
|
||||||
{
|
{
|
||||||
#ifdef DEBUG_HASHING
|
#ifdef DEBUG_HASHING
|
||||||
XP_U32 origHash = stack_getHash( stack );
|
XP_U32 origHash = stack_getHash( stack, XP_TRUE );
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
pushEntryImpl( stack, entry );
|
pushEntryImpl( stack, entry );
|
||||||
|
@ -341,7 +253,7 @@ pushEntry( StackCtxt* stack, const StackEntry* entry )
|
||||||
XP_U32 newHash = stack_getHash( stack );
|
XP_U32 newHash = stack_getHash( stack );
|
||||||
StackEntry lastEntry;
|
StackEntry lastEntry;
|
||||||
if ( stack_popEntry( stack, &lastEntry ) ) {
|
if ( stack_popEntry( stack, &lastEntry ) ) {
|
||||||
XP_ASSERT( origHash == stack_getHash( stack ) );
|
XP_ASSERT( origHash == stack_getHash( stack, XP_TRUE ) );
|
||||||
pushEntryImpl( stack, &lastEntry );
|
pushEntryImpl( stack, &lastEntry );
|
||||||
XP_ASSERT( newHash == stack_getHash( stack ) );
|
XP_ASSERT( newHash == stack_getHash( stack ) );
|
||||||
XP_LOGF( "%s: all ok", __func__ );
|
XP_LOGF( "%s: all ok", __func__ );
|
||||||
|
|
|
@ -69,8 +69,7 @@ StackCtxt* stack_make( MPFORMAL VTableMgr* vtmgr );
|
||||||
void stack_destroy( StackCtxt* stack );
|
void stack_destroy( StackCtxt* stack );
|
||||||
|
|
||||||
void stack_init( StackCtxt* stack );
|
void stack_init( StackCtxt* stack );
|
||||||
XP_U32 stack_getHashOld( StackCtxt* stack );
|
XP_U32 stack_getHash( const StackCtxt* stack, XP_Bool correct );
|
||||||
XP_U32 stack_getHash( const StackCtxt* stack );
|
|
||||||
void stack_setBitsPerTile( StackCtxt* stack, XP_U16 bitsPerTile );
|
void stack_setBitsPerTile( StackCtxt* stack, XP_U16 bitsPerTile );
|
||||||
|
|
||||||
void stack_loadFromStream( StackCtxt* stack, XWStreamCtxt* stream );
|
void stack_loadFromStream( StackCtxt* stack, XWStreamCtxt* stream );
|
||||||
|
|
|
@ -2032,7 +2032,7 @@ sendMoveTo( ServerCtxt* server, XP_U16 devIndex, XP_U16 turn,
|
||||||
XP_U16 version = stream_getVersion( stream );
|
XP_U16 version = stream_getVersion( stream );
|
||||||
if ( STREAM_VERS_BIGBOARD <= version ) {
|
if ( STREAM_VERS_BIGBOARD <= version ) {
|
||||||
XP_ASSERT( version == server->nv.streamVersion );
|
XP_ASSERT( version == server->nv.streamVersion );
|
||||||
XP_U32 hash = model_getHash( server->vol.model, version );
|
XP_U32 hash = model_getHash( server->vol.model );
|
||||||
#ifdef DEBUG_HASHING
|
#ifdef DEBUG_HASHING
|
||||||
XP_LOGF( "%s: adding hash %x", __func__, (unsigned int)hash );
|
XP_LOGF( "%s: adding hash %x", __func__, (unsigned int)hash );
|
||||||
#endif
|
#endif
|
||||||
|
|
|
@ -228,6 +228,31 @@ p_replaceStringIfDifferent( MPFORMAL XP_UCHAR** curLoc, const XP_UCHAR* newStr
|
||||||
*curLoc = curStr;
|
*curLoc = curStr;
|
||||||
} /* replaceStringIfDifferent */
|
} /* replaceStringIfDifferent */
|
||||||
|
|
||||||
|
XP_U32
|
||||||
|
augmentHash( XP_U32 hash, const XP_U8* ptr, XP_U16 len )
|
||||||
|
{
|
||||||
|
XP_ASSERT( 0 < len );
|
||||||
|
// see http://en.wikipedia.org/wiki/Jenkins_hash_function
|
||||||
|
for ( XP_U16 ii = 0; ii < len; ++ii ) {
|
||||||
|
hash += *ptr++;
|
||||||
|
hash += (hash << 10);
|
||||||
|
hash ^= (hash >> 6);
|
||||||
|
}
|
||||||
|
#ifdef DEBUG_HASHING
|
||||||
|
XP_LOGF( "%s: hashed %d bytes -> %X", __func__, len, (unsigned int)hash );
|
||||||
|
#endif
|
||||||
|
return hash;
|
||||||
|
}
|
||||||
|
|
||||||
|
XP_U32
|
||||||
|
finishHash( XP_U32 hash )
|
||||||
|
{
|
||||||
|
hash += (hash << 3);
|
||||||
|
hash ^= (hash >> 11);
|
||||||
|
hash += (hash << 15);
|
||||||
|
return hash;
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* A wrapper for printing etc. potentially null strings.
|
* A wrapper for printing etc. potentially null strings.
|
||||||
*/
|
*/
|
||||||
|
|
|
@ -68,6 +68,8 @@ XP_UCHAR* p_copyString( MPFORMAL const XP_UCHAR* instr
|
||||||
# define copyString( p, in ) p_copyString( in )
|
# define copyString( p, in ) p_copyString( in )
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
XP_U32 augmentHash( XP_U32 hash, const XP_U8* ptr, XP_U16 len );
|
||||||
|
XP_U32 finishHash( XP_U32 hash );
|
||||||
|
|
||||||
void p_replaceStringIfDifferent( MPFORMAL XP_UCHAR** curLoc,
|
void p_replaceStringIfDifferent( MPFORMAL XP_UCHAR** curLoc,
|
||||||
const XP_UCHAR* newStr
|
const XP_UCHAR* newStr
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
/* -*-mode: C; fill-column: 78; c-basic-offset: 4; -*- */
|
/* -*-mode: C; fill-column: 78; c-basic-offset: 4; -*- */
|
||||||
/*
|
/*
|
||||||
* Copyright 1997 - 2000 by Eric House (xwords@eehouse.org). All rights reserved.
|
* Copyright 1997 - 2015 by Eric House (xwords@eehouse.org). All rights
|
||||||
|
* reserved.
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or
|
* This program is free software; you can redistribute it and/or
|
||||||
* modify it under the terms of the GNU General Public License
|
* modify it under the terms of the GNU General Public License
|
||||||
|
@ -48,7 +49,7 @@ typedef struct StreamCtxVTable {
|
||||||
XP_U16 (*m_stream_getU16)( XWStreamCtxt* dctx );
|
XP_U16 (*m_stream_getU16)( XWStreamCtxt* dctx );
|
||||||
XP_U32 (*m_stream_getU32)( XWStreamCtxt* dctx );
|
XP_U32 (*m_stream_getU32)( XWStreamCtxt* dctx );
|
||||||
XP_U32 (*m_stream_getBits)( XWStreamCtxt* dctx, XP_U16 nBits );
|
XP_U32 (*m_stream_getBits)( XWStreamCtxt* dctx, XP_U16 nBits );
|
||||||
#if defined HASH_STREAM || defined DEBUG
|
#if defined DEBUG
|
||||||
void (*m_stream_copyBits)( const XWStreamCtxt* dctx, XWStreamPos endPos,
|
void (*m_stream_copyBits)( const XWStreamCtxt* dctx, XWStreamPos endPos,
|
||||||
XP_U8* buf, XP_U16* len );
|
XP_U8* buf, XP_U16* len );
|
||||||
#endif
|
#endif
|
||||||
|
@ -73,6 +74,8 @@ typedef struct StreamCtxVTable {
|
||||||
void (*m_stream_close)( XWStreamCtxt* dctx );
|
void (*m_stream_close)( XWStreamCtxt* dctx );
|
||||||
|
|
||||||
XP_U16 (*m_stream_getSize)( const XWStreamCtxt* dctx );
|
XP_U16 (*m_stream_getSize)( const XWStreamCtxt* dctx );
|
||||||
|
XP_U32 (*m_stream_getHash)( const XWStreamCtxt* dctx, XWStreamPos pos,
|
||||||
|
XP_Bool correct );
|
||||||
|
|
||||||
const XP_U8* (*m_stream_getPtr)( const XWStreamCtxt* dctx );
|
const XP_U8* (*m_stream_getPtr)( const XWStreamCtxt* dctx );
|
||||||
|
|
||||||
|
@ -113,7 +116,7 @@ struct XWStreamCtxt {
|
||||||
#define stream_getBits(sc, n) \
|
#define stream_getBits(sc, n) \
|
||||||
(sc)->vtable->m_stream_getBits((sc), (n))
|
(sc)->vtable->m_stream_getBits((sc), (n))
|
||||||
|
|
||||||
#if defined HASH_STREAM || defined DEBUG
|
#if defined DEBUG
|
||||||
#define stream_copyBits(sc, e, b, l) \
|
#define stream_copyBits(sc, e, b, l) \
|
||||||
(sc)->vtable->m_stream_copyBits((sc), (e), (b), (l))
|
(sc)->vtable->m_stream_copyBits((sc), (e), (b), (l))
|
||||||
#endif
|
#endif
|
||||||
|
@ -154,6 +157,9 @@ struct XWStreamCtxt {
|
||||||
#define stream_getSize(sc) \
|
#define stream_getSize(sc) \
|
||||||
(sc)->vtable->m_stream_getSize((sc))
|
(sc)->vtable->m_stream_getSize((sc))
|
||||||
|
|
||||||
|
#define stream_getHash(sc, p, c) \
|
||||||
|
(sc)->vtable->m_stream_getHash((sc), (p), (c))
|
||||||
|
|
||||||
#define stream_getPtr(sc) \
|
#define stream_getPtr(sc) \
|
||||||
(sc)->vtable->m_stream_getPtr((sc))
|
(sc)->vtable->m_stream_getPtr((sc))
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue