annotate src/lstream.h @ 665:fdefd0186b75

[xemacs-hg @ 2001-09-20 06:28:42 by ben] The great integral types renaming. The purpose of this is to rationalize the names used for various integral types, so that they match their intended uses and follow consist conventions, and eliminate types that were not semantically different from each other. The conventions are: -- All integral types that measure quantities of anything are signed. Some people disagree vociferously with this, but their arguments are mostly theoretical, and are vastly outweighed by the practical headaches of mixing signed and unsigned values, and more importantly by the far increased likelihood of inadvertent bugs: Because of the broken "viral" nature of unsigned quantities in C (operations involving mixed signed/unsigned are done unsigned, when exactly the opposite is nearly always wanted), even a single error in declaring a quantity unsigned that should be signed, or even the even more subtle error of comparing signed and unsigned values and forgetting the necessary cast, can be catastrophic, as comparisons will yield wrong results. -Wsign-compare is turned on specifically to catch this, but this tends to result in a great number of warnings when mixing signed and unsigned, and the casts are annoying. More has been written on this elsewhere. -- All such quantity types just mentioned boil down to EMACS_INT, which is 32 bits on 32-bit machines and 64 bits on 64-bit machines. This is guaranteed to be the same size as Lisp objects of type `int', and (as far as I can tell) of size_t (unsigned!) and ssize_t. The only type below that is not an EMACS_INT is Hashcode, which is an unsigned value of the same size as EMACS_INT. -- Type names should be relatively short (no more than 10 characters or so), with the first letter capitalized and no underscores if they can at all be avoided. -- "count" == a zero-based measurement of some quantity. Includes sizes, offsets, and indexes. -- "bpos" == a one-based measurement of a position in a buffer. "Charbpos" and "Bytebpos" count text in the buffer, rather than bytes in memory; thus Bytebpos does not directly correspond to the memory representation. Use "Membpos" for this. -- "Char" refers to internal-format characters, not to the C type "char", which is really a byte. -- For the actual name changes, see the script below. I ran the following script to do the conversion. (NOTE: This script is idempotent. You can safely run it multiple times and it will not screw up previous results -- in fact, it will do nothing if nothing has changed. Thus, it can be run repeatedly as necessary to handle patches coming in from old workspaces, or old branches.) There are two tags, just before and just after the change: `pre-integral-type-rename' and `post-integral-type-rename'. When merging code from the main trunk into a branch, the best thing to do is first merge up to `pre-integral-type-rename', then apply the script and associated changes, then merge from `post-integral-type-change' to the present. (Alternatively, just do the merging in one operation; but you may then have a lot of conflicts needing to be resolved by hand.) Script `fixtypes.sh' follows: ----------------------------------- cut ------------------------------------ files="*.[ch] s/*.h m/*.h config.h.in ../configure.in Makefile.in.in ../lib-src/*.[ch] ../lwlib/*.[ch]" gr Memory_Count Bytecount $files gr Lstream_Data_Count Bytecount $files gr Element_Count Elemcount $files gr Hash_Code Hashcode $files gr extcount bytecount $files gr bufpos charbpos $files gr bytind bytebpos $files gr memind membpos $files gr bufbyte intbyte $files gr Extcount Bytecount $files gr Bufpos Charbpos $files gr Bytind Bytebpos $files gr Memind Membpos $files gr Bufbyte Intbyte $files gr EXTCOUNT BYTECOUNT $files gr BUFPOS CHARBPOS $files gr BYTIND BYTEBPOS $files gr MEMIND MEMBPOS $files gr BUFBYTE INTBYTE $files gr MEMORY_COUNT BYTECOUNT $files gr LSTREAM_DATA_COUNT BYTECOUNT $files gr ELEMENT_COUNT ELEMCOUNT $files gr HASH_CODE HASHCODE $files ----------------------------------- cut ------------------------------------ `fixtypes.sh' is a Bourne-shell script; it uses 'gr': ----------------------------------- cut ------------------------------------ #!/bin/sh # Usage is like this: # gr FROM TO FILES ... # globally replace FROM with TO in FILES. FROM and TO are regular expressions. # backup files are stored in the `backup' directory. from="$1" to="$2" shift 2 echo ${1+"$@"} | xargs global-replace "s/$from/$to/g" ----------------------------------- cut ------------------------------------ `gr' in turn uses a Perl script to do its real work, `global-replace', which follows: ----------------------------------- cut ------------------------------------ : #-*- Perl -*- ### global-modify --- modify the contents of a file by a Perl expression ## Copyright (C) 1999 Martin Buchholz. ## Copyright (C) 2001 Ben Wing. ## Authors: Martin Buchholz <martin@xemacs.org>, Ben Wing <ben@xemacs.org> ## Maintainer: Ben Wing <ben@xemacs.org> ## Current Version: 1.0, May 5, 2001 # 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 XEmacs; see the file COPYING. If not, write to the Free # Software Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA # 02111-1307, USA. eval 'exec perl -w -S $0 ${1+"$@"}' if 0; use strict; use FileHandle; use Carp; use Getopt::Long; use File::Basename; (my $myName = $0) =~ s@.*/@@; my $usage=" Usage: $myName [--help] [--backup-dir=DIR] [--line-mode] [--hunk-mode] PERLEXPR FILE ... Globally modify a file, either line by line or in one big hunk. Typical usage is like this: [with GNU print, GNU xargs: guaranteed to handle spaces, quotes, etc. in file names] find . -name '*.[ch]' -print0 | xargs -0 $0 's/\bCONST\b/const/g'\n [with non-GNU print, xargs] find . -name '*.[ch]' -print | xargs $0 's/\bCONST\b/const/g'\n The file is read in, either line by line (with --line-mode specified) or in one big hunk (with --hunk-mode specified; it's the default), and the Perl expression is then evalled with \$_ set to the line or hunk of text, including the terminating newline if there is one. It should destructively modify the value there, storing the changed result in \$_. Files in which any modifications are made are backed up to the directory specified using --backup-dir, or to `backup' by default. To disable this, use --backup-dir= with no argument. Hunk mode is the default because it is MUCH MUCH faster than line-by-line. Use line-by-line only when it matters, e.g. you want to do a replacement only once per line (the default without the `g' argument). Conversely, when using hunk mode, *ALWAYS* use `g'; otherwise, you will only make one replacement in the entire file! "; my %options = (); $Getopt::Long::ignorecase = 0; &GetOptions ( \%options, 'help', 'backup-dir=s', 'line-mode', 'hunk-mode', ); die $usage if $options{"help"} or @ARGV <= 1; my $code = shift; die $usage if grep (-d || ! -w, @ARGV); sub SafeOpen { open ((my $fh = new FileHandle), $_[0]); confess "Can't open $_[0]: $!" if ! defined $fh; return $fh; } sub SafeClose { close $_[0] or confess "Can't close $_[0]: $!"; } sub FileContents { my $fh = SafeOpen ("< $_[0]"); my $olddollarslash = $/; local $/ = undef; my $contents = <$fh>; $/ = $olddollarslash; return $contents; } sub WriteStringToFile { my $fh = SafeOpen ("> $_[0]"); binmode $fh; print $fh $_[1] or confess "$_[0]: $!\n"; SafeClose $fh; } foreach my $file (@ARGV) { my $changed_p = 0; my $new_contents = ""; if ($options{"line-mode"}) { my $fh = SafeOpen $file; while (<$fh>) { my $save_line = $_; eval $code; $changed_p = 1 if $save_line ne $_; $new_contents .= $_; } } else { my $orig_contents = $_ = FileContents $file; eval $code; if ($_ ne $orig_contents) { $changed_p = 1; $new_contents = $_; } } if ($changed_p) { my $backdir = $options{"backup-dir"}; $backdir = "backup" if !defined ($backdir); if ($backdir) { my ($name, $path, $suffix) = fileparse ($file, ""); my $backfulldir = $path . $backdir; my $backfile = "$backfulldir/$name"; mkdir $backfulldir, 0755 unless -d $backfulldir; print "modifying $file (original saved in $backfile)\n"; rename $file, $backfile; } WriteStringToFile ($file, $new_contents); } } ----------------------------------- cut ------------------------------------ In addition to those programs, I needed to fix up a few other things, particularly relating to the duplicate definitions of types, now that some types merged with others. Specifically: 1. in lisp.h, removed duplicate declarations of Bytecount. The changed code should now look like this: (In each code snippet below, the first and last lines are the same as the original, as are all lines outside of those lines. That allows you to locate the section to be replaced, and replace the stuff in that section, verifying that there isn't anything new added that would need to be kept.) --------------------------------- snip ------------------------------------- /* Counts of bytes or chars */ typedef EMACS_INT Bytecount; typedef EMACS_INT Charcount; /* Counts of elements */ typedef EMACS_INT Elemcount; /* Hash codes */ typedef unsigned long Hashcode; /* ------------------------ dynamic arrays ------------------- */ --------------------------------- snip ------------------------------------- 2. in lstream.h, removed duplicate declaration of Bytecount. Rewrote the comment about this type. The changed code should now look like this: --------------------------------- snip ------------------------------------- #endif /* The have been some arguments over the what the type should be that specifies a count of bytes in a data block to be written out or read in, using Lstream_read(), Lstream_write(), and related functions. Originally it was long, which worked fine; Martin "corrected" these to size_t and ssize_t on the grounds that this is theoretically cleaner and is in keeping with the C standards. Unfortunately, this practice is horribly error-prone due to design flaws in the way that mixed signed/unsigned arithmetic happens. In fact, by doing this change, Martin introduced a subtle but fatal error that caused the operation of sending large mail messages to the SMTP server under Windows to fail. By putting all values back to be signed, avoiding any signed/unsigned mixing, the bug immediately went away. The type then in use was Lstream_Data_Count, so that it be reverted cleanly if a vote came to that. Now it is Bytecount. Some earlier comments about why the type must be signed: This MUST BE SIGNED, since it also is used in functions that return the number of bytes actually read to or written from in an operation, and these functions can return -1 to signal error. Note that the standard Unix read() and write() functions define the count going in as a size_t, which is UNSIGNED, and the count going out as an ssize_t, which is SIGNED. This is a horrible design flaw. Not only is it highly likely to lead to logic errors when a -1 gets interpreted as a large positive number, but operations are bound to fail in all sorts of horrible ways when a number in the upper-half of the size_t range is passed in -- this number is unrepresentable as an ssize_t, so code that checks to see how many bytes are actually written (which is mandatory if you are dealing with certain types of devices) will get completely screwed up. --ben */ typedef enum lstream_buffering --------------------------------- snip ------------------------------------- 3. in dumper.c, there are four places, all inside of switch() statements, where XD_BYTECOUNT appears twice as a case tag. In each case, the two case blocks contain identical code, and you should *REMOVE THE SECOND* and leave the first.
author ben
date Thu, 20 Sep 2001 06:31:11 +0000
parents b39c14581166
children 943eaba38521
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
428
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
1 /* Generic stream implementation -- header file.
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
2 Copyright (C) 1995 Free Software Foundation, Inc.
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
3 Copyright (C) 1996 Ben Wing.
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
4
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
5 This file is part of XEmacs.
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
6
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
7 XEmacs is free software; you can redistribute it and/or modify it
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
8 under the terms of the GNU General Public License as published by the
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
9 Free Software Foundation; either version 2, or (at your option) any
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
10 later version.
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
11
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
12 XEmacs is distributed in the hope that it will be useful, but WITHOUT
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
13 ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
14 FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
15 for more details.
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
16
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
17 You should have received a copy of the GNU General Public License
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
18 along with XEmacs; see the file COPYING. If not, write to
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
19 the Free Software Foundation, Inc., 59 Temple Place - Suite 330,
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
20 Boston, MA 02111-1307, USA. */
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
21
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
22 /* Synched up with: Not in FSF. */
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
23
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
24 /* Written by Ben Wing. */
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
25
440
8de8e3f6228a Import from CVS: tag r21-2-28
cvs
parents: 428
diff changeset
26 #ifndef INCLUDED_lstream_h_
8de8e3f6228a Import from CVS: tag r21-2-28
cvs
parents: 428
diff changeset
27 #define INCLUDED_lstream_h_
428
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
28
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
29 /************************************************************************/
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
30 /* definition of Lstream object */
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
31 /************************************************************************/
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
32
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
33 DECLARE_LRECORD (lstream, struct lstream);
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
34 #define XLSTREAM(x) XRECORD (x, lstream, struct lstream)
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
35 #define XSETLSTREAM(x, p) XSETRECORD (x, p, lstream)
617
af57a77cbc92 [xemacs-hg @ 2001-06-18 07:09:50 by ben]
ben
parents: 563
diff changeset
36 #define wrap_lstream(p) wrap_record (p, lstream)
428
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
37 #define LSTREAMP(x) RECORDP (x, lstream)
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
38 /* #define CHECK_LSTREAM(x) CHECK_RECORD (x, lstream)
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
39 Lstream pointers should never escape to the Lisp level, so
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
40 functions should not be doing this. */
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
41
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
42 #ifndef EOF
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
43 #define EOF (-1)
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
44 #endif
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
45
665
fdefd0186b75 [xemacs-hg @ 2001-09-20 06:28:42 by ben]
ben
parents: 647
diff changeset
46 /* The have been some arguments over the what the type should be that
fdefd0186b75 [xemacs-hg @ 2001-09-20 06:28:42 by ben]
ben
parents: 647
diff changeset
47 specifies a count of bytes in a data block to be written out or read in,
fdefd0186b75 [xemacs-hg @ 2001-09-20 06:28:42 by ben]
ben
parents: 647
diff changeset
48 using Lstream_read(), Lstream_write(), and related functions.
fdefd0186b75 [xemacs-hg @ 2001-09-20 06:28:42 by ben]
ben
parents: 647
diff changeset
49 Originally it was long, which worked fine; Martin "corrected" these to
fdefd0186b75 [xemacs-hg @ 2001-09-20 06:28:42 by ben]
ben
parents: 647
diff changeset
50 size_t and ssize_t on the grounds that this is theoretically cleaner and
fdefd0186b75 [xemacs-hg @ 2001-09-20 06:28:42 by ben]
ben
parents: 647
diff changeset
51 is in keeping with the C standards. Unfortunately, this practice is
fdefd0186b75 [xemacs-hg @ 2001-09-20 06:28:42 by ben]
ben
parents: 647
diff changeset
52 horribly error-prone due to design flaws in the way that mixed
fdefd0186b75 [xemacs-hg @ 2001-09-20 06:28:42 by ben]
ben
parents: 647
diff changeset
53 signed/unsigned arithmetic happens. In fact, by doing this change,
fdefd0186b75 [xemacs-hg @ 2001-09-20 06:28:42 by ben]
ben
parents: 647
diff changeset
54 Martin introduced a subtle but fatal error that caused the operation of
fdefd0186b75 [xemacs-hg @ 2001-09-20 06:28:42 by ben]
ben
parents: 647
diff changeset
55 sending large mail messages to the SMTP server under Windows to fail.
fdefd0186b75 [xemacs-hg @ 2001-09-20 06:28:42 by ben]
ben
parents: 647
diff changeset
56 By putting all values back to be signed, avoiding any signed/unsigned
fdefd0186b75 [xemacs-hg @ 2001-09-20 06:28:42 by ben]
ben
parents: 647
diff changeset
57 mixing, the bug immediately went away. The type then in use was
fdefd0186b75 [xemacs-hg @ 2001-09-20 06:28:42 by ben]
ben
parents: 647
diff changeset
58 Lstream_Data_Count, so that it be reverted cleanly if a vote came to
fdefd0186b75 [xemacs-hg @ 2001-09-20 06:28:42 by ben]
ben
parents: 647
diff changeset
59 that. Now it is Bytecount.
fdefd0186b75 [xemacs-hg @ 2001-09-20 06:28:42 by ben]
ben
parents: 647
diff changeset
60
fdefd0186b75 [xemacs-hg @ 2001-09-20 06:28:42 by ben]
ben
parents: 647
diff changeset
61 Some earlier comments about why the type must be signed: This MUST BE
fdefd0186b75 [xemacs-hg @ 2001-09-20 06:28:42 by ben]
ben
parents: 647
diff changeset
62 SIGNED, since it also is used in functions that return the number of
fdefd0186b75 [xemacs-hg @ 2001-09-20 06:28:42 by ben]
ben
parents: 647
diff changeset
63 bytes actually read to or written from in an operation, and these
fdefd0186b75 [xemacs-hg @ 2001-09-20 06:28:42 by ben]
ben
parents: 647
diff changeset
64 functions can return -1 to signal error.
462
0784d089fdc9 Import from CVS: tag r21-2-46
cvs
parents: 456
diff changeset
65
0784d089fdc9 Import from CVS: tag r21-2-46
cvs
parents: 456
diff changeset
66 Note that the standard Unix read() and write() functions define the
0784d089fdc9 Import from CVS: tag r21-2-46
cvs
parents: 456
diff changeset
67 count going in as a size_t, which is UNSIGNED, and the count going
0784d089fdc9 Import from CVS: tag r21-2-46
cvs
parents: 456
diff changeset
68 out as an ssize_t, which is SIGNED. This is a horrible design
0784d089fdc9 Import from CVS: tag r21-2-46
cvs
parents: 456
diff changeset
69 flaw. Not only is it highly likely to lead to logic errors when a
0784d089fdc9 Import from CVS: tag r21-2-46
cvs
parents: 456
diff changeset
70 -1 gets interpreted as a large positive number, but operations are
0784d089fdc9 Import from CVS: tag r21-2-46
cvs
parents: 456
diff changeset
71 bound to fail in all sorts of horrible ways when a number in the
0784d089fdc9 Import from CVS: tag r21-2-46
cvs
parents: 456
diff changeset
72 upper-half of the size_t range is passed in -- this number is
0784d089fdc9 Import from CVS: tag r21-2-46
cvs
parents: 456
diff changeset
73 unrepresentable as an ssize_t, so code that checks to see how many
0784d089fdc9 Import from CVS: tag r21-2-46
cvs
parents: 456
diff changeset
74 bytes are actually written (which is mandatory if you are dealing
0784d089fdc9 Import from CVS: tag r21-2-46
cvs
parents: 456
diff changeset
75 with certain types of devices) will get completely screwed up.
665
fdefd0186b75 [xemacs-hg @ 2001-09-20 06:28:42 by ben]
ben
parents: 647
diff changeset
76
fdefd0186b75 [xemacs-hg @ 2001-09-20 06:28:42 by ben]
ben
parents: 647
diff changeset
77 --ben
462
0784d089fdc9 Import from CVS: tag r21-2-46
cvs
parents: 456
diff changeset
78 */
0784d089fdc9 Import from CVS: tag r21-2-46
cvs
parents: 456
diff changeset
79
428
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
80 typedef enum lstream_buffering
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
81 {
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
82 /* No buffering. */
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
83 LSTREAM_UNBUFFERED,
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
84 /* Buffer until a '\n' character is reached. */
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
85 LSTREAM_LINE_BUFFERED,
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
86 /* Buffer in standard-size (i.e. 512-byte) blocks. */
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
87 LSTREAM_BLOCK_BUFFERED,
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
88 /* Buffer in blocks of a specified size. */
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
89 LSTREAM_BLOCKN_BUFFERED,
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
90 /* Buffer until the stream is closed (only applies to write-only
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
91 streams). Only one call to the stream writer will be made,
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
92 and that is when the stream is closed. */
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
93 LSTREAM_UNLIMITED
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
94 } Lstream_buffering;
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
95
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
96 /* Methods defining how this stream works. Some may be undefined. */
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
97
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
98 /* We do not implement the seek/tell paradigm. I tried to do that,
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
99 but getting the semantics right in the presence of buffering is
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
100 extremely tricky and very error-prone and basically not worth it.
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
101 This is especially the case with complicated streams like
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
102 decoding streams -- the seek pointer in this case can't be a single
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
103 integer but has to be a whole complicated structure that records
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
104 all of the stream's state at the time.
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
105
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
106 Rewind semantics are generally easy to implement, so we do provide
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
107 a rewind method. Even rewind() may not be available on a stream,
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
108 however -- e.g. on process output. */
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
109
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
110 typedef struct lstream_implementation
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
111 {
442
abe6d1db359e Import from CVS: tag r21-2-36
cvs
parents: 440
diff changeset
112 const char *name;
665
fdefd0186b75 [xemacs-hg @ 2001-09-20 06:28:42 by ben]
ben
parents: 647
diff changeset
113 Bytecount size; /* Number of additional bytes to be
462
0784d089fdc9 Import from CVS: tag r21-2-46
cvs
parents: 456
diff changeset
114 allocated with this stream. Access this
0784d089fdc9 Import from CVS: tag r21-2-46
cvs
parents: 456
diff changeset
115 data using Lstream_data(). */
428
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
116 /* Read some data from the stream's end and store it into DATA, which
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
117 can hold SIZE bytes. Return the number of bytes read. A return
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
118 value of 0 means no bytes can be read at this time. This may
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
119 be because of an EOF, or because there is a granularity greater
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
120 than one byte that the stream imposes on the returned data, and
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
121 SIZE is less than this granularity. (This will happen frequently
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
122 for streams that need to return whole characters, because
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
123 Lstream_read() calls the reader function repeatedly until it
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
124 has the number of bytes it wants or until 0 is returned.)
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
125 The lstream functions do not treat a 0 return as EOF or do
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
126 anything special; however, the calling function will interpret
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
127 any 0 it gets back as EOF. This will normally not happen unless
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
128 the caller calls Lstream_read() with a very small size.
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
129
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
130 This function can be NULL if the stream is output-only. */
442
abe6d1db359e Import from CVS: tag r21-2-36
cvs
parents: 440
diff changeset
131 /* The omniscient mly, blinded by the irresistible thrall of Common
428
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
132 Lisp, thinks that it is bogus that the types and implementations
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
133 of input and output streams are the same. */
665
fdefd0186b75 [xemacs-hg @ 2001-09-20 06:28:42 by ben]
ben
parents: 647
diff changeset
134 Bytecount (*reader) (Lstream *stream, unsigned char *data,
fdefd0186b75 [xemacs-hg @ 2001-09-20 06:28:42 by ben]
ben
parents: 647
diff changeset
135 Bytecount size);
428
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
136 /* Send some data to the stream's end. Data to be sent is in DATA
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
137 and is SIZE bytes. Return the number of bytes sent. This
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
138 function can send and return fewer bytes than is passed in; in
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
139 that case, the function will just be called again until there is
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
140 no data left or 0 is returned. A return value of 0 means that no
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
141 more data can be currently stored, but there is no error; the
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
142 data will be squirrelled away until the writer can accept
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
143 data. (This is useful, e.g., of you're dealing with a
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
144 non-blocking file descriptor and are getting EWOULDBLOCK errors.)
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
145 This function can be NULL if the stream is input-only. */
665
fdefd0186b75 [xemacs-hg @ 2001-09-20 06:28:42 by ben]
ben
parents: 647
diff changeset
146 Bytecount (*writer) (Lstream *stream, const unsigned char *data,
fdefd0186b75 [xemacs-hg @ 2001-09-20 06:28:42 by ben]
ben
parents: 647
diff changeset
147 Bytecount size);
428
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
148 /* Return non-zero if the last write operation on the stream resulted
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
149 in an attempt to block (EWOULDBLOCK). If this method does not
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
150 exists, the implementation returns 0 */
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
151 int (*was_blocked_p) (Lstream *stream);
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
152 /* Rewind the stream. If this is NULL, the stream is not seekable. */
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
153 int (*rewinder) (Lstream *stream);
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
154 /* Indicate whether this stream is seekable -- i.e. it can be rewound.
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
155 This method is ignored if the stream does not have a rewind
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
156 method. If this method is not present, the result is determined
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
157 by whether a rewind method is present. */
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
158 int (*seekable_p) (Lstream *stream);
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
159 /* Perform any additional operations necessary to flush the
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
160 data in this stream. */
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
161 int (*flusher) (Lstream *stream);
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
162 /* Perform any additional operations necessary to close this
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
163 stream down. May be NULL. This function is called when
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
164 Lstream_close() is called or when the stream is garbage-
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
165 collected. When this function is called, all pending data
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
166 in the stream will already have been written out. */
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
167 int (*closer) (Lstream *stream);
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
168 /* Mark this object for garbage collection. Same semantics as
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
169 a standard Lisp_Object marker. This function can be NULL. */
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
170 Lisp_Object (*marker) (Lisp_Object lstream);
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
171 } Lstream_implementation;
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
172
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
173 #define DEFINE_LSTREAM_IMPLEMENTATION(name,c_name,size) \
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
174 Lstream_implementation c_name[1] = \
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
175 { { (name), (size) } }
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
176
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
177 #define LSTREAM_FL_IS_OPEN 1
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
178 #define LSTREAM_FL_READ 2
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
179 #define LSTREAM_FL_WRITE 4
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
180 #define LSTREAM_FL_NO_PARTIAL_CHARS 8
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
181 #define LSTREAM_FL_CLOSE_AT_DISKSAVE 16
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
182
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
183 struct lstream
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
184 {
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
185 struct lcrecord_header header;
442
abe6d1db359e Import from CVS: tag r21-2-36
cvs
parents: 440
diff changeset
186 const Lstream_implementation *imp; /* methods for this stream */
428
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
187 Lstream_buffering buffering; /* type of buffering in use */
665
fdefd0186b75 [xemacs-hg @ 2001-09-20 06:28:42 by ben]
ben
parents: 647
diff changeset
188 Bytecount buffering_size; /* number of bytes buffered */
428
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
189
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
190 unsigned char *in_buffer; /* holds characters read from stream end */
665
fdefd0186b75 [xemacs-hg @ 2001-09-20 06:28:42 by ben]
ben
parents: 647
diff changeset
191 Bytecount in_buffer_size; /* allocated size of buffer */
fdefd0186b75 [xemacs-hg @ 2001-09-20 06:28:42 by ben]
ben
parents: 647
diff changeset
192 Bytecount in_buffer_current; /* number of characters in buffer */
fdefd0186b75 [xemacs-hg @ 2001-09-20 06:28:42 by ben]
ben
parents: 647
diff changeset
193 Bytecount in_buffer_ind; /* pointer to next character to
462
0784d089fdc9 Import from CVS: tag r21-2-46
cvs
parents: 456
diff changeset
194 take from buffer */
428
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
195
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
196 unsigned char *out_buffer; /* holds characters to write to stream end */
665
fdefd0186b75 [xemacs-hg @ 2001-09-20 06:28:42 by ben]
ben
parents: 647
diff changeset
197 Bytecount out_buffer_size; /* allocated size of buffer */
fdefd0186b75 [xemacs-hg @ 2001-09-20 06:28:42 by ben]
ben
parents: 647
diff changeset
198 Bytecount out_buffer_ind; /* pointer to next buffer spot to
462
0784d089fdc9 Import from CVS: tag r21-2-46
cvs
parents: 456
diff changeset
199 write a character */
428
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
200
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
201 /* The unget buffer is more or less a stack -- things get pushed
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
202 onto the end and read back from the end. Lstream_read()
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
203 basically reads backwards from the end to get stuff; Lstream_unread()
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
204 similarly has to push the data on backwards. */
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
205 unsigned char *unget_buffer; /* holds characters pushed back onto input */
665
fdefd0186b75 [xemacs-hg @ 2001-09-20 06:28:42 by ben]
ben
parents: 647
diff changeset
206 Bytecount unget_buffer_size; /* allocated size of buffer */
fdefd0186b75 [xemacs-hg @ 2001-09-20 06:28:42 by ben]
ben
parents: 647
diff changeset
207 Bytecount unget_buffer_ind; /* pointer to next buffer spot
462
0784d089fdc9 Import from CVS: tag r21-2-46
cvs
parents: 456
diff changeset
208 to write a character */
428
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
209
665
fdefd0186b75 [xemacs-hg @ 2001-09-20 06:28:42 by ben]
ben
parents: 647
diff changeset
210 Bytecount byte_count;
456
e7ef97881643 Import from CVS: tag r21-2-43
cvs
parents: 444
diff changeset
211 int flags;
e7ef97881643 Import from CVS: tag r21-2-43
cvs
parents: 444
diff changeset
212 max_align_t data[1];
428
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
213 };
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
214
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
215 #define LSTREAM_TYPE_P(lstr, type) \
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
216 ((lstr)->imp == lstream_##type)
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
217
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
218 #ifdef ERROR_CHECK_TYPECHECK
442
abe6d1db359e Import from CVS: tag r21-2-36
cvs
parents: 440
diff changeset
219 INLINE_HEADER struct lstream *
428
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
220 error_check_lstream_type (struct lstream *stream,
442
abe6d1db359e Import from CVS: tag r21-2-36
cvs
parents: 440
diff changeset
221 const Lstream_implementation *imp);
abe6d1db359e Import from CVS: tag r21-2-36
cvs
parents: 440
diff changeset
222 INLINE_HEADER struct lstream *
428
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
223 error_check_lstream_type (struct lstream *stream,
442
abe6d1db359e Import from CVS: tag r21-2-36
cvs
parents: 440
diff changeset
224 const Lstream_implementation *imp)
428
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
225 {
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
226 assert (stream->imp == imp);
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
227 return stream;
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
228 }
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
229 # define LSTREAM_TYPE_DATA(lstr, type) \
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
230 ((struct type##_stream *) \
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
231 Lstream_data (error_check_lstream_type(lstr, lstream_##type)))
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
232 #else
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
233 # define LSTREAM_TYPE_DATA(lstr, type) \
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
234 ((struct type##_stream *) Lstream_data (lstr))
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
235 #endif
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
236
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
237 /* Declare that lstream-type TYPE has method M; used in
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
238 initialization routines */
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
239 #define LSTREAM_HAS_METHOD(type, m) \
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
240 (lstream_##type->m = type##_##m)
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
241
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
242
442
abe6d1db359e Import from CVS: tag r21-2-36
cvs
parents: 440
diff changeset
243 Lstream *Lstream_new (const Lstream_implementation *imp,
abe6d1db359e Import from CVS: tag r21-2-36
cvs
parents: 440
diff changeset
244 const char *mode);
428
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
245 void Lstream_reopen (Lstream *lstr);
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
246 void Lstream_set_buffering (Lstream *lstr, Lstream_buffering buffering,
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
247 int buffering_size);
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
248 int Lstream_flush (Lstream *lstr);
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
249 int Lstream_flush_out (Lstream *lstr);
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
250 int Lstream_fputc (Lstream *lstr, int c);
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
251 int Lstream_fgetc (Lstream *lstr);
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
252 void Lstream_fungetc (Lstream *lstr, int c);
665
fdefd0186b75 [xemacs-hg @ 2001-09-20 06:28:42 by ben]
ben
parents: 647
diff changeset
253 Bytecount Lstream_read (Lstream *lstr, void *data,
fdefd0186b75 [xemacs-hg @ 2001-09-20 06:28:42 by ben]
ben
parents: 647
diff changeset
254 Bytecount size);
fdefd0186b75 [xemacs-hg @ 2001-09-20 06:28:42 by ben]
ben
parents: 647
diff changeset
255 Bytecount Lstream_write (Lstream *lstr, const void *data,
fdefd0186b75 [xemacs-hg @ 2001-09-20 06:28:42 by ben]
ben
parents: 647
diff changeset
256 Bytecount size);
428
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
257 int Lstream_was_blocked_p (Lstream *lstr);
665
fdefd0186b75 [xemacs-hg @ 2001-09-20 06:28:42 by ben]
ben
parents: 647
diff changeset
258 void Lstream_unread (Lstream *lstr, const void *data, Bytecount size);
428
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
259 int Lstream_rewind (Lstream *lstr);
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
260 int Lstream_seekable_p (Lstream *lstr);
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
261 int Lstream_close (Lstream *lstr);
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
262 void Lstream_delete (Lstream *lstr);
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
263 void Lstream_set_character_mode (Lstream *str);
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
264
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
265 /* Call the function equivalent if the out buffer is full. Otherwise,
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
266 add to the end of the out buffer and, if line buffering is called for
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
267 and the character marks the end of a line, write out the buffer. */
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
268
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
269 #define Lstream_putc(stream, c) \
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
270 ((stream)->out_buffer_ind >= (stream)->out_buffer_size ? \
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
271 Lstream_fputc (stream, c) : \
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
272 ((stream)->out_buffer[(stream)->out_buffer_ind++] = \
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
273 (unsigned char) (c), \
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
274 (stream)->byte_count++, \
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
275 (stream)->buffering == LSTREAM_LINE_BUFFERED && \
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
276 (stream)->out_buffer[(stream)->out_buffer_ind - 1] == '\n' ? \
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
277 Lstream_flush_out (stream) : 0))
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
278
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
279 /* Retrieve from unget buffer if there are any characters there;
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
280 else retrieve from in buffer if there's anything there;
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
281 else call the function equivalent */
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
282 #define Lstream_getc(stream) \
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
283 ((stream)->unget_buffer_ind > 0 ? \
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
284 ((stream)->byte_count++, \
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
285 (stream)->unget_buffer[--(stream)->unget_buffer_ind]) : \
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
286 (stream)->in_buffer_ind < (stream)->in_buffer_current ? \
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
287 ((stream)->byte_count++, \
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
288 (stream)->in_buffer[(stream)->in_buffer_ind++]) : \
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
289 Lstream_fgetc (stream))
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
290
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
291 /* Add to the end if it won't overflow buffer; otherwise call the
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
292 function equivalent */
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
293 #define Lstream_ungetc(stream, c) \
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
294 ((stream)->unget_buffer_ind >= (stream)->unget_buffer_size ? \
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
295 Lstream_fungetc (stream, c) : \
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
296 (void) ((stream)->byte_count--, \
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
297 ((stream)->unget_buffer[(stream)->unget_buffer_ind++] = \
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
298 (unsigned char) (c))))
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
299
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
300 #define Lstream_data(stream) ((void *) ((stream)->data))
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
301 #define Lstream_byte_count(stream) ((stream)->byte_count)
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
302
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
303
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
304 /************************************************************************/
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
305 /* working with an Lstream as a stream of Emchars */
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
306 /************************************************************************/
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
307
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
308 #ifdef MULE
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
309
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
310 #ifndef BYTE_ASCII_P
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
311 #include "mule-charset.h"
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
312 #endif
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
313
442
abe6d1db359e Import from CVS: tag r21-2-36
cvs
parents: 440
diff changeset
314 INLINE_HEADER Emchar Lstream_get_emchar (Lstream *stream);
abe6d1db359e Import from CVS: tag r21-2-36
cvs
parents: 440
diff changeset
315 INLINE_HEADER Emchar
428
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
316 Lstream_get_emchar (Lstream *stream)
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
317 {
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
318 int c = Lstream_getc (stream);
444
576fb035e263 Import from CVS: tag r21-2-37
cvs
parents: 442
diff changeset
319 return (c < 0x80 /* c == EOF || BYTE_ASCII_P (c) */
576fb035e263 Import from CVS: tag r21-2-37
cvs
parents: 442
diff changeset
320 ? (Emchar) c
576fb035e263 Import from CVS: tag r21-2-37
cvs
parents: 442
diff changeset
321 : Lstream_get_emchar_1 (stream, c));
428
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
322 }
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
323
442
abe6d1db359e Import from CVS: tag r21-2-36
cvs
parents: 440
diff changeset
324 INLINE_HEADER int Lstream_put_emchar (Lstream *stream, Emchar ch);
abe6d1db359e Import from CVS: tag r21-2-36
cvs
parents: 440
diff changeset
325 INLINE_HEADER int
428
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
326 Lstream_put_emchar (Lstream *stream, Emchar ch)
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
327 {
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
328 return CHAR_ASCII_P (ch) ?
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
329 Lstream_putc (stream, ch) :
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
330 Lstream_fput_emchar (stream, ch);
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
331 }
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
332
442
abe6d1db359e Import from CVS: tag r21-2-36
cvs
parents: 440
diff changeset
333 INLINE_HEADER void Lstream_unget_emchar (Lstream *stream, Emchar ch);
abe6d1db359e Import from CVS: tag r21-2-36
cvs
parents: 440
diff changeset
334 INLINE_HEADER void
428
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
335 Lstream_unget_emchar (Lstream *stream, Emchar ch)
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
336 {
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
337 if (CHAR_ASCII_P (ch))
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
338 Lstream_ungetc (stream, ch);
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
339 else
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
340 Lstream_funget_emchar (stream, ch);
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
341 }
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
342 #else /* not MULE */
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
343
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
344 # define Lstream_get_emchar(stream) Lstream_getc (stream)
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
345 # define Lstream_put_emchar(stream, ch) Lstream_putc (stream, ch)
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
346 # define Lstream_unget_emchar(stream, ch) Lstream_ungetc (stream, ch)
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
347
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
348 #endif /* not MULE */
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
349
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
350
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
351 /************************************************************************/
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
352 /* Lstream implementations */
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
353 /************************************************************************/
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
354
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
355 /* Flags we can pass to the filedesc and stdio streams. */
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
356
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
357 /* If set, close the descriptor or FILE * when the stream is closed. */
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
358 #define LSTR_CLOSING 1
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
359
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
360 /* If set, allow quitting out of the actual I/O. */
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
361 #define LSTR_ALLOW_QUIT 2
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
362
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
363 /* If set and filedesc_stream_set_pty_flushing() has been called
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
364 on the stream, do not send more than pty_max_bytes on a single
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
365 line without flushing the data out using the eof_char. */
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
366 #define LSTR_PTY_FLUSHING 4
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
367
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
368 /* If set, an EWOULDBLOCK error is not treated as an error but
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
369 simply causes the write function to return 0 as the number
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
370 of bytes written out. */
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
371 #define LSTR_BLOCKED_OK 8
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
372
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
373 Lisp_Object make_stdio_input_stream (FILE *stream, int flags);
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
374 Lisp_Object make_stdio_output_stream (FILE *stream, int flags);
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
375 Lisp_Object make_filedesc_input_stream (int filedesc, int offset, int count,
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
376 int flags);
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
377 Lisp_Object make_filedesc_output_stream (int filedesc, int offset, int count,
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
378 int flags);
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
379 void filedesc_stream_set_pty_flushing (Lstream *stream,
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
380 int pty_max_bytes,
665
fdefd0186b75 [xemacs-hg @ 2001-09-20 06:28:42 by ben]
ben
parents: 647
diff changeset
381 Intbyte eof_char);
428
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
382 int filedesc_stream_fd (Lstream *stream);
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
383 Lisp_Object make_lisp_string_input_stream (Lisp_Object string,
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
384 Bytecount offset,
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
385 Bytecount len);
462
0784d089fdc9 Import from CVS: tag r21-2-46
cvs
parents: 456
diff changeset
386 Lisp_Object make_fixed_buffer_input_stream (const void *buf,
665
fdefd0186b75 [xemacs-hg @ 2001-09-20 06:28:42 by ben]
ben
parents: 647
diff changeset
387 Bytecount size);
462
0784d089fdc9 Import from CVS: tag r21-2-46
cvs
parents: 456
diff changeset
388 Lisp_Object make_fixed_buffer_output_stream (void *buf,
665
fdefd0186b75 [xemacs-hg @ 2001-09-20 06:28:42 by ben]
ben
parents: 647
diff changeset
389 Bytecount size);
442
abe6d1db359e Import from CVS: tag r21-2-36
cvs
parents: 440
diff changeset
390 const unsigned char *fixed_buffer_input_stream_ptr (Lstream *stream);
428
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
391 unsigned char *fixed_buffer_output_stream_ptr (Lstream *stream);
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
392 Lisp_Object make_resizing_buffer_output_stream (void);
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
393 unsigned char *resizing_buffer_stream_ptr (Lstream *stream);
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
394 Lisp_Object make_dynarr_output_stream (unsigned_char_dynarr *dyn);
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
395 #define LSTR_SELECTIVE 1
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
396 #define LSTR_IGNORE_ACCESSIBLE 2
665
fdefd0186b75 [xemacs-hg @ 2001-09-20 06:28:42 by ben]
ben
parents: 647
diff changeset
397 Lisp_Object make_lisp_buffer_input_stream (struct buffer *buf, Charbpos start,
fdefd0186b75 [xemacs-hg @ 2001-09-20 06:28:42 by ben]
ben
parents: 647
diff changeset
398 Charbpos end, int flags);
fdefd0186b75 [xemacs-hg @ 2001-09-20 06:28:42 by ben]
ben
parents: 647
diff changeset
399 Lisp_Object make_lisp_buffer_output_stream (struct buffer *buf, Charbpos pos,
428
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
400 int flags);
665
fdefd0186b75 [xemacs-hg @ 2001-09-20 06:28:42 by ben]
ben
parents: 647
diff changeset
401 Charbpos lisp_buffer_stream_startpos (Lstream *stream);
428
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
402
440
8de8e3f6228a Import from CVS: tag r21-2-28
cvs
parents: 428
diff changeset
403 #endif /* INCLUDED_lstream_h_ */