* harbour/contrib/gd/gdwrp.c
* harbour/contrib/libmisc/hb_f.c
* harbour/contrib/samples/dbf.c
* harbour/contrib/samples/time.c
* harbour/doc/es/hb_apiln.txt
* harbour/include/hbapicdp.h
* harbour/include/hbapilng.h
* harbour/source/rdd/dbcmd.c
* harbour/source/rdd/dbnubs.c
* harbour/source/rdd/hsx/cftsfunc.c
* harbour/source/rtl/accept.c
* harbour/source/rtl/defpath.c
* harbour/source/rtl/oemansi.c
* harbour/source/rtl/pad.c
* harbour/source/rtl/shadow.c
* harbour/source/rtl/trim.c
* harbour/source/vm/debug.c
* changed HB_FUNCNAME( <name> )() to HB_FUNC_EXEC( <name> )
* harbour/utils/hbrun/Makefile
* changed order of linked libraries
* harbour/source/rtl/empty.c
* changed EMPTY( <symboItem> ) - not it returns .T. when <symboItem>
is not a function symbol
* harbour/source/rtl/errorapi.c
% replaced all hb_dynsym{Find,Get}*() by static variables initialized
at HVM startup variables - it reduce the cost of creating and
initializing error object about 50%
* harbour/include/hbapifs.h
* harbour/source/rtl/fssize.c
* harbour/source/rtl/filesys.c
* harbour/source/rtl/hbffind.c
* changed to use 64bit file API
+ harbour/contrib/libct/ctextern.ch
+ added list with EXTERN requests for our CT3 function. List generated
automatically from final binary CT lib created on Linux.
391 lines
9.7 KiB
C
391 lines
9.7 KiB
C
/*
|
|
* $Id$
|
|
*/
|
|
|
|
/*
|
|
* Harbour Project source code:
|
|
* File handling functions
|
|
*
|
|
* Copyright 1999 Andi Jahja <andij@aonlippo.co.id>
|
|
* www - http://www.harbour-project.org
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2, 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 General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this software; see the file COPYING. If not, write to
|
|
* the Free Software Foundation, Inc., 59 Temple Place, Suite 330,
|
|
* Boston, MA 02111-1307 USA (or visit the web site http://www.gnu.org/).
|
|
*
|
|
* As a special exception, the Harbour Project gives permission for
|
|
* additional uses of the text contained in its release of Harbour.
|
|
*
|
|
* The exception is that, if you link the Harbour libraries with other
|
|
* files to produce an executable, this does not by itself cause the
|
|
* resulting executable to be covered by the GNU General Public License.
|
|
* Your use of that executable is in no way restricted on account of
|
|
* linking the Harbour library code into it.
|
|
*
|
|
* This exception does not however invalidate any other reasons why
|
|
* the executable file might be covered by the GNU General Public License.
|
|
*
|
|
* This exception applies only to the code released by the Harbour
|
|
* Project under the name Harbour. If you copy code from other
|
|
* Harbour Project or Free Software Foundation releases into a copy of
|
|
* Harbour, as the General Public License permits, the exception does
|
|
* not apply to the code that you add in this way. To avoid misleading
|
|
* anyone as to the status of such modified files, you must delete
|
|
* this exception notice from them.
|
|
*
|
|
* If you write modifications of your own for Harbour, it is your choice
|
|
* whether to permit this exception to apply to your modifications.
|
|
* If you do not wish that, delete this exception notice.
|
|
*
|
|
*/
|
|
|
|
/* please run $(HARBOUR)\tests\testhbf.prg for testing */
|
|
|
|
#include "hbapifs.h"
|
|
|
|
#define b_size 4096
|
|
#define c_size 4096
|
|
|
|
static long hb_hbfskip( int recs );
|
|
|
|
static long last_rec[10];
|
|
static long recno[10];
|
|
static long offset[10];
|
|
static int handles[10];
|
|
static int area = 0;
|
|
static char *b;
|
|
static char *c;
|
|
static long last_off[10];
|
|
static long lastbyte[10];
|
|
static int isEof[10];
|
|
|
|
HB_FUNC( HB_FUSE )
|
|
{
|
|
|
|
PHB_ITEM arg1_it = hb_param(1,HB_IT_STRING);
|
|
PHB_ITEM arg2_it = hb_param(2,HB_IT_NUMERIC);
|
|
int open_flags;
|
|
|
|
if ( arg1_it ) {
|
|
|
|
if( arg2_it )
|
|
open_flags = hb_parni(2);
|
|
else
|
|
open_flags = 0;
|
|
|
|
handles[area] = hb_fsOpen( ( BYTE * ) hb_parc(1), open_flags );
|
|
offset[area] = 0;
|
|
recno[area] = 1;
|
|
b = ( char * )hb_xgrab( b_size );
|
|
c = ( char * )hb_xgrab( c_size );
|
|
lastbyte[area] = hb_fsSeek( handles[area], 0L, SEEK_END );
|
|
isEof[area] = (lastbyte[area] == 0);
|
|
hb_retni( handles[area] );
|
|
}
|
|
else {
|
|
hb_fsClose( handles[area] );
|
|
hb_xfree( b ) ;
|
|
hb_xfree( c ) ;
|
|
hb_retni( 1 ) ;
|
|
recno[area] = 0L ;
|
|
offset[area] = 0L ;
|
|
handles[area] = 0 ;
|
|
last_rec[area] = 0L ;
|
|
last_off[area] = 0L ;
|
|
lastbyte[area] = 0L ;
|
|
isEof[area] = 0 ;
|
|
}
|
|
}
|
|
|
|
|
|
HB_FUNC( HB_FRECNO )
|
|
{
|
|
hb_retnl( recno[area] );
|
|
}
|
|
|
|
|
|
HB_FUNC( HB_FSKIP )
|
|
{
|
|
|
|
PHB_ITEM arg1_it = hb_param(1,HB_IT_NUMERIC);
|
|
int nskip;
|
|
|
|
if( arg1_it )
|
|
nskip = hb_parni(1);
|
|
else
|
|
nskip = 1;
|
|
|
|
hb_hbfskip(nskip);
|
|
|
|
}
|
|
|
|
static long hb_hbfskip( int recs )
|
|
{
|
|
long read_pos;
|
|
long read_len;
|
|
long x, y;
|
|
|
|
|
|
HB_TRACE(HB_TR_DEBUG, ("hb_hbskip(%d)", recs));
|
|
|
|
if ( recs > 0 ) {
|
|
for (y = 0; y < recs; y++ ) {
|
|
hb_fsSeek( handles[area], offset[area], SEEK_SET );
|
|
read_len = hb_fsRead( handles[area], ( BYTE * ) b, b_size );
|
|
for (x = 0; x < read_len; x++ ) {
|
|
if ( ((*(b + x) == 13) && (*(b + x + 1) == 10)) ||
|
|
((*(b + x) == 10) && (*(b + x + 1) == 13)) ) {
|
|
break;
|
|
}
|
|
}
|
|
if ( (offset[area] + x + 2) < lastbyte[area] ) {
|
|
isEof[area] = FALSE;
|
|
offset[area] += (x + 2);
|
|
recno[area] += 1;
|
|
}
|
|
else
|
|
isEof[area] = TRUE;
|
|
}
|
|
}
|
|
else {
|
|
recs = -recs;
|
|
isEof[area] = FALSE;
|
|
|
|
if ( (recno[area] - recs) < 1 )
|
|
return( 1 );
|
|
|
|
for (y = recs; y > 0; y-- ) {
|
|
if ( offset[area] - b_size < 0L ) {
|
|
read_pos = 0;
|
|
read_len = (size_t)offset[area];
|
|
}
|
|
else {
|
|
read_pos = (size_t)(offset[area] - b_size);
|
|
read_len = b_size;
|
|
}
|
|
|
|
hb_fsSeek( handles[area], read_pos, SEEK_SET );
|
|
read_len = hb_fsRead( handles[area], ( BYTE * ) b, ( USHORT )read_len );
|
|
|
|
for (x = read_len - 4; x >= 0; x-- ) {
|
|
if ( ((*(b + x) == 13) && (*(b + x + 1) == 10)) ||
|
|
((*(b + x) == 10) && (*(b + x + 1) == 13)) ) {
|
|
break;
|
|
}
|
|
}
|
|
if ( x < 0 ) {
|
|
offset[area] = 0;
|
|
recno[area] = 1;
|
|
}
|
|
else {
|
|
offset[area] = read_pos + x + 2;
|
|
recno[area]--;
|
|
}
|
|
}
|
|
}
|
|
|
|
return ( recno[area] );
|
|
}
|
|
|
|
HB_FUNC( HB_FREADLN )
|
|
{
|
|
|
|
int x;
|
|
long read;
|
|
|
|
hb_fsSeek( handles[area], offset[area], SEEK_SET );
|
|
read = hb_fsRead( handles[area], ( BYTE * ) b, b_size );
|
|
|
|
for ( x = 0; x < b_size; x++ ) {
|
|
if ( ((*(b + x) == 13) && (*(b + x + 1) == 10)) ||
|
|
((*(b + x) == 10) && (*(b + x + 1) == 13)) ||
|
|
(*(b + x) == 26) || ( x >= (int)read) ) {
|
|
break;
|
|
}
|
|
}
|
|
hb_retclen( b, x );
|
|
|
|
}
|
|
|
|
HB_FUNC( HB_FEOF )
|
|
{
|
|
hb_retl( isEof[area] );
|
|
}
|
|
|
|
HB_FUNC( HB_FGOTO )
|
|
{
|
|
|
|
long target;
|
|
long last;
|
|
|
|
target = hb_parnl(1);
|
|
|
|
if ( recno[area] > target ) {
|
|
while ( recno[area] != target ) {
|
|
last = recno[area];
|
|
hb_hbfskip(-1);
|
|
if ( recno[area] == last )
|
|
break;
|
|
}
|
|
}
|
|
else {
|
|
while ( recno[area] != target ) {
|
|
last = recno[area];
|
|
hb_hbfskip(1);
|
|
if ( recno[area] == last )
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
HB_FUNC( HB_FGOBOTTOM )
|
|
{
|
|
int x;
|
|
int len;
|
|
long loc, last;
|
|
|
|
if ( last_rec[area] != 0 ) {
|
|
recno[area] = last_rec[area];
|
|
offset[area] = last_off[area];
|
|
}
|
|
else {
|
|
|
|
loc = 0L;
|
|
|
|
last = offset[area];
|
|
do {
|
|
hb_fsSeek( handles[area], offset[area], SEEK_SET );
|
|
len = hb_fsRead( handles[area], ( BYTE * ) c, c_size );
|
|
for ( x = 0; x < len; x++ ) {
|
|
if ( ((*(c + x) == 13) && (*(c + x + 1) == 10)) ||
|
|
((*(c + x) == 10) && (*(c + x + 1) == 13)) ||
|
|
( x - loc > b_size ) ) {
|
|
last = offset[area] + loc;
|
|
recno[area]++;
|
|
x++;
|
|
loc = x + 1;
|
|
}
|
|
}
|
|
offset[area] += loc;
|
|
|
|
} while ( len == c_size );
|
|
|
|
last_rec[area] = --recno[area];
|
|
last_off[area] = last;
|
|
}
|
|
}
|
|
|
|
HB_FUNC( HB_FGOTOP )
|
|
{
|
|
offset[area] = 0L;
|
|
recno[area] = 1L;
|
|
isEof[area] = (lastbyte[area] == 0);
|
|
}
|
|
|
|
HB_FUNC( HB_FLASTREC )
|
|
{
|
|
long old_rec;
|
|
long old_offset;
|
|
int bIsEof;
|
|
|
|
old_rec = recno[area];
|
|
old_offset = offset[area];
|
|
bIsEof = isEof[area];
|
|
|
|
HB_FUNC_EXEC( HB_FGOBOTTOM );
|
|
hb_retnl( last_rec[area] );
|
|
|
|
recno[area] = old_rec;
|
|
offset[area] = old_offset;
|
|
isEof[area] = bIsEof ;
|
|
}
|
|
|
|
HB_FUNC( HB_FSELECT )
|
|
{
|
|
hb_retni( area + 1 );
|
|
|
|
if ( ISNUM(1) )
|
|
area = hb_parni(1) - 1;
|
|
}
|
|
|
|
HB_FUNC( HB_FINFO ) /* used for debugging */
|
|
{
|
|
hb_reta( 6 );
|
|
hb_storni( area+1, -1, 1);
|
|
hb_storni( last_rec[area], -1, 2);
|
|
hb_storni( recno[area], -1, 3);
|
|
hb_storni( offset[area], -1, 4);
|
|
hb_storni( lastbyte[area], -1, 5);
|
|
hb_storl ( isEof[area], -1, 6);
|
|
|
|
}
|
|
|
|
HB_FUNC( HB_FREADANDSKIP )
|
|
{
|
|
/* ------------------------------------------------
|
|
Warning: This is a rogue function! It is a first shot at adding the logic
|
|
to read .CSV records that respect CRLF embedded within quotes.
|
|
It is very common, especially with Microsoft products, for
|
|
comma-separated files to allow a field (usually an address field)
|
|
to have hard returns within it. These records appear corrupted to any
|
|
reader that presumes all hard returns are record separators.
|
|
|
|
This function is useful right now to loop through a CSV file
|
|
while !hb_feof(), but it does NOT recognize the same record count
|
|
and positioning that the other functions in this file use.
|
|
It does its own skip and read, so an entire file can be read
|
|
sequentially with just this function.
|
|
-BH
|
|
--------------------------------------------------*/
|
|
long x = 0;
|
|
long read;
|
|
BOOL bInField = 0, bHasCRLF = FALSE;
|
|
|
|
hb_fsSeek( handles[area], offset[area], SEEK_SET );
|
|
read = hb_fsRead( handles[area], ( BYTE * ) b, b_size );
|
|
|
|
while ( x < read )
|
|
{
|
|
if ( *(b + x) == '"' )
|
|
{
|
|
bInField = !bInField ;
|
|
x++;
|
|
continue;
|
|
}
|
|
if ( bInField )
|
|
{
|
|
x++;
|
|
continue;
|
|
}
|
|
if( ((*(b + x) == 13) && x < read-1 && (*(b + x + 1) == 10)) ||
|
|
((*(b + x) == 10) && x < read-1 && (*(b + x + 1) == 13)) )
|
|
{
|
|
x += 2;
|
|
bHasCRLF = TRUE;
|
|
break;
|
|
}
|
|
x++;
|
|
}
|
|
|
|
offset[area] = offset[area] + x;
|
|
recno[area] += 1;
|
|
// See if there's more to read
|
|
if ( !isEof[area] )
|
|
isEof[area] = (lastbyte[area] <= offset[area] + 1) ;
|
|
|
|
hb_retclen( b, x - (bHasCRLF ? 2 : 0) );
|
|
|
|
}
|
|
|