⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 write.c

📁 开放源码的编译器open watcom 1.6.0版的源代码
💻 C
字号:
/****************************************************************************
*
*                            Open Watcom Project
*
*    Portions Copyright (c) 1983-2002 Sybase, Inc. All Rights Reserved.
*
*  ========================================================================
*
*    This file contains Original Code and/or Modifications of Original
*    Code as defined in and that are subject to the Sybase Open Watcom
*    Public License version 1.0 (the 'License'). You may not use this file
*    except in compliance with the License. BY USING THIS FILE YOU AGREE TO
*    ALL TERMS AND CONDITIONS OF THE LICENSE. A copy of the License is
*    provided with the Original Code and Modifications, and is also
*    available at www.sybase.com/developer/opensource.
*
*    The Original Code and all software distributed under the License are
*    distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER
*    EXPRESS OR IMPLIED, AND SYBASE AND ALL CONTRIBUTORS HEREBY DISCLAIM
*    ALL SUCH WARRANTIES, INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF
*    MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR
*    NON-INFRINGEMENT. Please see the License for the specific language
*    governing rights and limitations under the License.
*
*  ========================================================================
*
* Description:  WHEN YOU FIGURE OUT WHAT THIS FILE DOES, PLEASE
*               DESCRIBE IT HERE!
*
****************************************************************************/


#include <string.h>
#include "filefmt.h"
#include "resfmt.h"
#include "mresfmt.h"
#include "wresrtns.h"
#include "layer0.h"
#include "write.h"
#include "reserr.h"
#include "read.h"

#define CONV_BUF_SIZE           512

static char     ConvBuffer[ CONV_BUF_SIZE ];

static int DefaultConversion( int len, const char *str, char *buf ) {
    int         i;

    if( buf != NULL ) {
        for( i=0; i < len; i++ ) {
            buf[ 2 * i ] = str[i];
            buf[ 2 * i + 1 ] = 0;
        }
    }
    return( len * 2 );
}

int (*ConvToUnicode)( int, const char *, char *) = DefaultConversion;

extern int ResWriteUint8( const uint_8 *newint, WResFileID handle )
/******************************************************************/
{
    int numwrite;

    numwrite = (* WRESWRITE) ( handle, newint, sizeof(uint_8) );
    if( numwrite != sizeof(uint_8) ) {
        WRES_ERROR( WRS_WRITE_FAILED );
        return( TRUE );
    } else {
        return( FALSE );
    }
}

int ResWriteUint16( const uint_16 *newint, WResFileID handle )
/*************************************************************/
{
    int numwrite;

    numwrite = (* WRESWRITE) ( handle, newint, sizeof( uint_16 ) );
    if( numwrite != sizeof(uint_16) ) {
        WRES_ERROR( WRS_WRITE_FAILED );
        return( TRUE );
    } else {
        return( FALSE );
    }
}

int ResWriteUint32( const uint_32 *newint, WResFileID handle )
/*************************************************************/
{
    int numwrite;

    numwrite = (* WRESWRITE) ( handle, newint, sizeof( uint_32 ) );
    if( numwrite != sizeof( uint_32 ) ) {
        WRES_ERROR( WRS_WRITE_FAILED );
        return( TRUE );
    } else {
        return( FALSE );
    }
}

int WResWriteWResIDNameUni( const WResIDName *name, uint_8 use_unicode,
                                    WResFileID handle )
/**********************************************************************/
{
    int         error;
    int         numwrote;
    uint_16     numchars;
    uint_8      tmp;
    char        *ptr;
    int         freebuf;

    freebuf = FALSE;
    error = FALSE;
    if (name == NULL) {
        /* a NULL name means write 0 length name */
        numchars = 0;
    } else {
        numchars = name->NumChars;
    }
    if( use_unicode ) {
        // for short strings use a static buffer in improve performance
        if( numchars <= CONV_BUF_SIZE / 2 ) {
            ptr = ConvBuffer;
        } else {
            freebuf = TRUE;
            ptr = WRESALLOC( 2 * numchars );
        }
        numchars = (ConvToUnicode)( numchars, name->Name, ptr ) / 2;
        error = ResWriteUint16( &numchars, handle );
        numchars *= 2;
    } else {
        numchars &= 0xFF;       /* in 16-bit the string can be no more than
                                   256 characters*/
        tmp = numchars;
        ptr = (char *)name->Name;
        error = ResWriteUint8( &tmp, handle );
    }
    if( !error && numchars > 0 ) {
        numwrote = (* WRESWRITE) ( handle, ptr, numchars );
        if( numwrote != numchars ) {
            error = TRUE;
            WRES_ERROR( WRS_WRITE_FAILED );
        }
    }
    if( freebuf ) {
        WRESFREE( ptr );
    }
    return( error );
} /* WResWriteWResIDNameUni */

int WResWriteWResIDName( const WResIDName *name, WResFileID handle )
/*******************************************************************/
{
    return( WResWriteWResIDNameUni( name, FALSE, handle ) );
}

int WResWriteWResID( const WResID *name, WResFileID handle )
/***********************************************************/
{
    int         error;
    uint_16     tmp16;
    uint_8      tmp8;

    tmp8 = name->IsName;
    error = ResWriteUint8( &tmp8, handle );
    if( !error ) {
        if( name->IsName ) {
            error = WResWriteWResIDName( &(name->ID.Name), handle );
        } else {
            tmp16 = name->ID.Num;
            error = ResWriteUint16( &tmp16, handle );
        }
    }

    return( error );
} /* WResWriteWResID */

/*
 * WResWriteTypeRecord - write the type record to the current postion
 *                       in the file identified by fp
 */
int WResWriteTypeRecord( const WResTypeInfo *type, WResFileID handle )
{
    int     numwrote;
    int     size;

    if( type->TypeName.IsName ) {
        /* -1 because one of the chars in the name is declared in the struct */
        size = sizeof( WResTypeInfo ) + type->TypeName.ID.Name.NumChars - 1;
    } else {
        size = sizeof( WResTypeInfo );
    }

    numwrote = (* WRESWRITE) ( handle, type, size );

    if( numwrote != size ) {
        WRES_ERROR( WRS_WRITE_FAILED );
        return( TRUE );
    } else {
        return( FALSE );
    }
} /* WResWriteTypeRecord */

/*
 * WResWriteResRecord - write the resource record to the current position
 *                      in the file identified by  fp
 */
int WResWriteResRecord( const WResResInfo *res, WResFileID handle )
{
    int     numwrote;
    int     size;

    if (res->ResName.IsName) {
        /* -1 because one of the chars in the name is declared in the struct */
        size = sizeof(WResResInfo) + res->ResName.ID.Name.NumChars - 1;
    } else {
        size = sizeof(WResResInfo);
    }

    numwrote = (* WRESWRITE) ( handle, (uint_8 *)res, size );
    if( numwrote != size ) {
        WRES_ERROR( WRS_WRITE_FAILED );
        return( TRUE );
    } else {
        return( FALSE );
    }
} /* WResWriteResRecord */

/*
 * WResWriteLangRecord - write out a language record at the current file
 *                       position
 */
int WResWriteLangRecord( const WResLangInfo *info, WResFileID handle ) {

    int     numwrote;

    numwrote = (* WRESWRITE) ( handle, info, sizeof( WResLangInfo ) );
    if( numwrote != sizeof( WResLangInfo ) ) {
        WRES_ERROR( WRS_WRITE_FAILED );
        return( TRUE );
    } else {
        return( FALSE );
    }
}

int WResWriteHeaderRecord( const WResHeader * header, WResFileID handle )
/***********************************************************************/
{
    off_t   seekpos;
    int     error;
    int     numwrite;

    seekpos = (* WRESSEEK) ( handle, 0L, SEEK_SET );
    error = (seekpos == -1L);
    if( error ) {
        WRES_ERROR( WRS_SEEK_FAILED );
    }

    if (!error) {
        numwrite = (* WRESWRITE) ( handle, header, sizeof(WResHeader) );
        if( numwrite != sizeof( WResHeader ) ) {
            error = TRUE;
            WRES_ERROR( WRS_WRITE_FAILED );
        }
    }

    return( error );
} /* WResWriteHeaderRecord */

int WResWriteExtHeader( const WResExtHeader * ext_head, WResFileID handle )
/*************************************************************************/
{
    int     numwrote;

    numwrote = WRESWRITE( handle, ext_head, sizeof(WResExtHeader) );
    if( numwrote != sizeof( WResExtHeader ) ) {
        WRES_ERROR( WRS_WRITE_FAILED );
        return( TRUE );
    } else {
        return( FALSE );
    }
}

extern int ResWriteStringLen( char * string, uint_8 use_unicode,
                              WResFileID handle, uint_16 len )
/*****************************************************************/
{
    int     numwrote;
    char    *buf;
    int     ret;

    if( use_unicode ) {
        if( len * 2 > CONV_BUF_SIZE ) {
            buf = WRESALLOC( 2 * len );
        } else {
            buf = ConvBuffer;
        }
        len = (ConvToUnicode)( len, string, buf );
    } else {
        buf = string;
    }
    numwrote = (* WRESWRITE) ( handle, buf, len );
    if( numwrote != len ) {
        WRES_ERROR( WRS_WRITE_FAILED );
        ret = TRUE;
    } else {
        ret = FALSE;
    }
    if( buf != ConvBuffer && buf != string ) {
        WRESFREE( buf );
    }
    return( ret );
}

extern int ResWriteString( char * string, uint_8 use_unicode, WResFileID handle)
/******************************************************************************/
{
    int     stringlen;
    int     ret;

    /* if string is NULL output the null string */
    if (string == NULL) {
        string = "";
    }

    /* the +1 is so we will output the '\0' as well */
    stringlen = strlen( string ) + 1;
    ret = ResWriteStringLen( string, use_unicode, handle, stringlen );
    return( ret );
}

int ResWriteNameOrOrdinal( ResNameOrOrdinal * name, uint_8 use_unicode,
                                         WResFileID handle )
/*********************************************************************/
{
    int         error;
    uint_16     flag;
    uint_16     tmp16;
    uint_8      tmp8;

    if( name == NULL ) {
        error = ResWriteString( "", use_unicode, handle );
    } else {
        if (name->ord.fFlag == 0xff) {
            if( use_unicode ) {
                flag = 0xffff;
                error = ResWriteUint16( &flag, handle );
            } else {
                tmp8 = name->ord.fFlag;
                error = ResWriteUint8( &tmp8, handle );
            }
            if (!error) {
                tmp16 = name->ord.wOrdinalID;
                ResWriteUint16( &tmp16, handle );
            }
        } else {
            error = ResWriteString( name->name, use_unicode, handle );
        }
    }

    return( error );
} /* ResWriteNameOrOrdinal */

static long MResFindNameOrOrdSize( ResNameOrOrdinal *data, char use_unicode )
/***************************************************************************/
{
    long       size;

    if( data->ord.fFlag == 0xff ) {
        size = 4;
    } else {
        if( use_unicode ) {
            size = 2 * ( strlen( data->name ) + 1 );
        } else {
            size = strlen( data->name ) + 1;
        }
    }

    return( size );
}

static int MResFindHeaderSize( MResResourceHeader *header, char use_unicode )
/***************************************************************************/
{
    int     headersize;
    long    namesize;
    long    typesize;
    long    padding;

    headersize = sizeof( MResResourceHeader ) - 2 * sizeof( ResNameOrOrdinal * )
                 + sizeof( uint_32 );
    namesize = MResFindNameOrOrdSize( header->Name, use_unicode );
    typesize = MResFindNameOrOrdSize( header->Type, use_unicode );
    headersize += ( namesize + typesize );
    padding = RES_PADDING( typesize + namesize, sizeof(uint_32) );

    return( headersize + padding );
}

int MResWriteResourceHeader( MResResourceHeader * currhead, WResFileID handle,
                             char iswin32 )
/*****************************************************************************/
{
    int         error;
    uint_32     headersize;
    uint_16     tmp16;
    uint_32     tmp32;

    if( !iswin32 ) {
        error = ResWriteNameOrOrdinal( currhead->Type, FALSE, handle );
        if (!error) {
            error = ResWriteNameOrOrdinal( currhead->Name, FALSE, handle );
        }
        if (!error) {
            tmp16 = currhead->MemoryFlags;
            error = ResWriteUint16( &tmp16, handle );
        }
        if (!error) {
            tmp32 = currhead->Size;
            error = ResWriteUint32( &tmp32, handle );
        }
    } else {
        tmp32 = currhead->Size;
        error = ResWriteUint32( &tmp32, handle );
        if( !error ) {
            headersize = MResFindHeaderSize( currhead, TRUE );
            error = ResWriteUint32( &headersize, handle  );
        }
        if( !error ) {
            error = ResWriteNameOrOrdinal( currhead->Type, TRUE, handle );
        }
        if( !error ) {
            error = ResWriteNameOrOrdinal( currhead->Name, TRUE, handle );
        }
        if( !error ) {
            error = ResPadDWord( handle );
        }
        if( !error ) {
            tmp32 = currhead->DataVersion;
            error = ResWriteUint32( &tmp32, handle );
        }
        if( !error ) {
            tmp16 = currhead->MemoryFlags;
            error = ResWriteUint16( &tmp16, handle );
        }
        if( !error ) {
            tmp16 = currhead->LanguageId;
            error = ResWriteUint16( &tmp16, handle );
        }
        if( !error ) {
            tmp32 = currhead->Version;
            error = ResWriteUint32( &tmp32, handle );
        }
        if( !error ) {
            tmp32 = currhead->Characteristics;
            error = ResWriteUint32( &tmp32, handle );
        }
    }

    return( error );
} /* MResWriteResourceHeader */

extern void WriteInitStatics( void )
/**********************************/
{
    memset( ConvBuffer, 0, CONV_BUF_SIZE * sizeof( char ) );
}

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -