From bd543f030e7e435c2c6a6a7d52ad927ae97cd927 Mon Sep 17 00:00:00 2001 From: "Thomas G. Lane" Date: Fri, 13 Dec 1991 00:00:00 +0000 Subject: [PATCH] The Independent JPEG Group's JPEG software v2 --- CHANGELOG | 37 ++++ README | 352 ++++++++++++++--------------------- SETUP | 235 ++++++++++++++++++++++++ USAGE | 162 +++++++++++++++++ ansi2knr.c | 37 ++-- cjpeg.1 | 130 +++++++++++++ config.c | 403 +++++++++++++++++++++++++++++++++++++++++ djpeg.1 | 114 ++++++++++++ egetopt.c | 96 +++++----- jbsmooth.c | 2 +- jccolor.c | 9 +- jcdeflts.c | 51 +++--- jchuff.c | 63 ++++--- jcmain.c | 106 +++++++---- jconfig.h | 18 +- jcpipe.c | 8 +- jcsample.c | 4 +- jdcolor.c | 6 +- jddeflts.c | 117 ++++++++++++ jdhuff.c | 43 +++-- jdmain.c | 150 +++++++-------- jdpipe.c | 3 +- jerror.c | 6 +- jfwddct.c | 187 ++++++++++++------- jinclude.h | 37 +++- jpegdata.h | 110 ++++++----- jquant1.c | 27 +-- jrdgif.c | 18 +- jrdjfif.c | 37 ++-- jrdppm.c | 281 ++++++++++++++++++++++++---- jrdrle.c | 372 +++++++++++++++++++++++++++++++++++++ jrdtarga.c | 483 +++++++++++++++++++++++++++++++++++++++++++++++++ jrevdct.c | 178 ++++++++++++------ jversion.h | 14 ++ jvirtmem.c | 76 +++++++- jwrgif.c | 19 +- jwrjfif.c | 6 +- jwrppm.c | 10 + jwrrle.c | 231 +++++++++++++++++++++++ jwrtarga.c | 223 +++++++++++++++++++++++ makcjpeg.cf | 7 +- makcjpeg.lnk | 14 +- makcjpeg.lst | 4 + makdjpeg.cf | 8 +- makdjpeg.lnk | 17 +- makdjpeg.lst | 4 + makefile.amiga | 143 --------------- makefile.ansi | 140 ++++++++++++++ makefile.manx | 133 ++++++++++++++ makefile.mc5 | 132 +++++++++----- makefile.mc6 | 121 ++++++++----- makefile.pwc | 111 +++++++----- makefile.sas | 152 ++++++++++++++++ makefile.tc | 113 ++++++++++++ makefile.unix | 143 ++++++++------- makljpeg.cf | 11 +- testimg.jpg | Bin 3583 -> 4383 bytes testimg.ppm | Bin 49167 -> 46890 bytes testorig.jpg | Bin 2899 -> 4380 bytes 59 files changed, 4578 insertions(+), 1136 deletions(-) create mode 100644 CHANGELOG create mode 100644 SETUP create mode 100644 USAGE create mode 100644 cjpeg.1 create mode 100644 config.c create mode 100644 djpeg.1 create mode 100644 jddeflts.c create mode 100644 jrdrle.c create mode 100644 jrdtarga.c create mode 100644 jversion.h create mode 100644 jwrrle.c create mode 100644 jwrtarga.c create mode 100644 makcjpeg.lst create mode 100644 makdjpeg.lst delete mode 100644 makefile.amiga create mode 100644 makefile.ansi create mode 100644 makefile.manx create mode 100644 makefile.sas create mode 100644 makefile.tc diff --git a/CHANGELOG b/CHANGELOG new file mode 100644 index 000000000..7146f385e --- /dev/null +++ b/CHANGELOG @@ -0,0 +1,37 @@ +CHANGELOG for Independent JPEG Group's JPEG software + +Version 2 13-Dec-91 +-------------------- + +Documentation improved a little --- there are man pages now. +Installation instructions moved from README to a separate file SETUP. + +New program config.c is provided to help you get the configuration options +right. This should make installation a lot more foolproof. + +Sense of djpeg -D switch reversed: dithering is now ON by default. + +RLE image file support added (thanks to Mike Lijewski). + +Targa image file support added (thanks to Lee Crocker). + +PPM input now accepts all PPM and PGM files. + +Bug fix: on machines where 'int' is 16 bits, high-Q-setting JPEG files +were not decoded correctly. + +Numerous changes to improve portability. There should be few or no compiler +warnings now. + +Makefiles cleaned up; defaults now appropriate for production use rather than +debugging. + +Subroutine interface cleaned up. If you wrote code based on version 1's +jcmain/jdmain, you'll need to change it, but it should get a little shorter +and simpler. + + +Version 1 7-Oct-91 +-------------------- + +Initial public release. diff --git a/README b/README index 9a2a1bbac..524f32957 100644 --- a/README +++ b/README @@ -1,27 +1,27 @@ The Independent JPEG Group's JPEG software ========================================== -README for release of 7-Oct-91 -=============================== +README for release 2 of 13-Dec-91 +================================= -This distribution contains the first public release of the Independent JPEG +This distribution contains the second public release of the Independent JPEG Group's free JPEG software. You are welcome to redistribute this software and to use it for any purpose, subject to the conditions under LEGAL ISSUES, below. -This software is still undergoing revision. Updated versions may be obtained -by anonymous FTP to uunet.uu.net; look under directory /graphics/jpeg. This -particular version will be archived as jpegsrc.v1.tar.Z. If you don't have -access to Internet FTP, UUNET's archives are also available via UUCP; contact -postmaster@uunet.uu.net for information on retrieving files that way. +For installation instructions, see file SETUP; for usage instructions, see +file USAGE (or the cjpeg.1 and djpeg.1 manual pages). -Please report any problems with this software to jpeg-info@uunet.uu.net. +This software is still undergoing revision. Updated versions may be obtained +by FTP or UUCP to uunet.uu.net and other archive sites; see ARCHIVE LOCATIONS +below for details. If you intend to become a serious user of this software, please contact -jpeg-info@uunet to be added to our electronic mailing list. Then you'll be -notified of updates and have a chance to participate in discussions, etc. +jpeg-info@uunet.uu.net to be added to our electronic mailing list. Then +you'll be notified of updates and have a chance to participate in discussions, +etc. -This software is the work of Tom Lane, Philip Gladstone, Luis Ortiz, and other -members of the independent JPEG group. +This software is the work of Tom Lane, Philip Gladstone, Luis Ortiz, +Lee Crocker, and other members of the Independent JPEG Group. DISCLAIMER @@ -32,20 +32,23 @@ useful for anything, nor to be compatible with subsequent releases, nor to be an accurate implementation of the JPEG standard. (See LEGAL ISSUES for even more disclaimers.) +Please report any problems with this software to jpeg-info@uunet.uu.net. + WHAT'S HERE =========== This distribution contains software to implement JPEG image compression and -decompression. JPEG is a standardized compression method for full-color and -gray-scale images. JPEG is intended for "real-world" scenes; cartoons and -other non-realistic images are not its strong suit. JPEG is lossy, meaning -that the output image is not necessarily identical to the input image. Hence -you should not use JPEG if you have to have identical output bits. However, -on typical images of real-world scenes, very good compression levels can be -obtained with hardly any visible change, and amazingly high compression levels -can be obtained if you can tolerate a low-quality image. For more details, -see the references, or just experiment with various compression settings. +decompression. JPEG (pronounced "jay-peg") is a standardized compression +method for full-color and gray-scale images. JPEG is intended for +"real-world" scenes; cartoons and other non-realistic images are not its +strong suit. JPEG is lossy, meaning that the output image is not necessarily +identical to the input image. Hence you should not use JPEG if you have to +have identical output bits. However, on typical images of real-world scenes, +very good compression levels can be obtained with no visible change, and +amazingly high compression levels can be obtained if you can tolerate a +low-quality image. For more details, see the references, or just experiment +with various compression settings. The software implements JPEG baseline and extended-sequential compression processes. Provision is made for supporting all variants of these processes, @@ -67,9 +70,9 @@ the future. This software can be used on several levels: * As canned software for JPEG compression and decompression. Just edit the - Makefile and configuration files as needed (see SETUP), compile and go. + Makefile and configuration files as needed (see file SETUP), compile and go. Members of the independent JPEG group will improve the out-of-the-box - functionality as time goes on. + functionality and speed as time goes on. * As the basis for other JPEG programs. For example, you could incorporate the decompressor into a general image viewing package by replacing the @@ -77,7 +80,7 @@ This software can be used on several levels: specific hardware, you might want to replace some of the inner loops with assembly code. For a non-command-line-driven system, you might want a different user interface. (Members of the group will be producing Macintosh - and Amiga versions with appropriate user interfaces, for example.) + and Amiga versions with more appropriate user interfaces, for example.) * As a toolkit for experimentation with JPEG and JPEG-like algorithms. Most of the individual decisions you might want to mess with are packaged up into @@ -90,181 +93,85 @@ In particular, we welcome the use of this software as the basis for commercial products; no royalty is required. -SETUP -===== - -The installation process is not very automatic; you will need at least some -familiarity with C programming and program build procedures for your system. -(Volunteers to work on improving this situation are welcome. Also, we will -probably start distributing pre-built binaries for popular systems at some -point.) - -First, select a makefile and copy it to "Makefile". "makefile.unix" -is appropriate for most Unix and Unix-like systems. Special makefiles are -included for various PC compilers. If you don't see a makefile for your -system, we recommend starting from makefile.unix. - -Look over the Makefile and adjust options as needed. In particular, you'll -need to change the CC= and CFLAGS= definitions if you don't have gcc -(makefile.unix only). If you have a function-prototype-less compiler, be sure -to uncomment the .c.o rule and say "make ansi2knr". This will cause the -source files to be preprocessed to change our ANSI-style function definitions -to old-style definitions. (Thanks to Peter Deutsch of Aladdin Enterprises for -ansi2knr.) - -Also look over jconfig.h and adjust #defines as necessary. If you have an -ANSI-compliant C compiler (gcc for instance), no changes should be necessary -except perhaps for RIGHT_SHIFT_IS_UNSIGNED and TWO_FILE_COMMANDLINE. For -older compilers other mods may be needed, depending on what ANSI features are -supported. If you prefer, you can usually leave jconfig.h unmodified and add --D switches to the Makefile's CFLAGS= definition. - -Then say "make". - -If you have trouble with missing system include files or inclusion of the -wrong ones, you can fix it in jinclude.h. In particular, if you are using -gcc on a machine with non-ANSI system include files, you are likely to find -that jinclude.h tries to include the wrong files (because gcc defines -__STDC__). There's no good automatic solution to this, so you'll just have -to hand-edit jinclude.h. - -As a quick test of functionality we've included three sample files: - testorig.jpg same as blkint.jpg from JPEG validation floppy. - testimg.ppm output of djpeg testorig.jpg - testimg.jpg output of cjpeg testimg.ppm -The two .jpg files aren't identical due to different parameter choices (and -wouldn't be anyway, since JPEG is lossy). However, if you can generate -duplicates of testimg.ppm and testimg.jpg then you probably have a working -port. "make test" will perform the necessary comparisons (by generating -testout.ppm and testout.jpg and comparing these to testimg.*). NOTE: this -is far from an exhaustive test of the JPEG software; some modules, such as -color quantization and GIF I/O, are not exercised at all. It's just a quick -test to give you some confidence that you haven't missed something major. - -If you need to make a smaller version of the JPEG software, some optional -functions can be removed at compile time. See the xxx_SUPPORTED #defines -in jconfig.h. (Not a lot is actually removed right now, but as more optional -stuff gets added, this mechanism will start to make a difference.) - -If you want to incorporate the JPEG code as subroutines in a larger program, -we recommend that you make libjpeg.a. Then use the .h files and libjpeg.a as -your interface to the JPEG functions. Your surrounding program will have to -provide functionality similar to what's in jcmain.c or jdmain.c, and you may -want to replace jerror.c and possibly other modules depending on your needs. -See the "architecture" file for more info. If it seems to you that the system -structure doesn't accommodate what you want to do, please contact the authors. - -Special notes for Macintosh Think C users: If you have version 5.0 you should -be able to just turn on __STDC__ through the compiler switch that enables -that. With version 4.0 you must manually edit jconfig.h to define PROTO, -HAVE_UNSIGNED_CHAR, HAVE_UNSIGNED_SHORT, and const. (It seems to be safe to -just define __STDC__ to take care of the first three.) When setting up -project files, use the COBJECTS and DOBJECTS lists in makefile.unix as a guide -to which files need to be included, and add the ANSI and Unix C libraries in a -separate segment. You may need to divide the JPEG files into more than one -segment; you can do this pretty much as you please. - - -USAGE -===== - -The user interface is pretty minimal at this point. We haven't bothered to -generate manual-page files since the switches badly need redesign. At the -moment, things work like this: - -There are two programs, cjpeg to compress an image file into JPEG format, -and djpeg to decompress. - -On Unix systems, you say: - cjpeg [switches] [imagefile] >jpegfile - djpeg [switches] [jpegfile] >imagefile -The programs read the specified input file, or standard input if none is -named. They always write to standard output (with trace/error messages to -standard error). These conventions are handy for piping images between -programs. - -On PC, Macintosh, and Amiga systems, you say: - cjpeg [switches] imagefile jpegfile - djpeg [switches] jpegfile imagefile -i.e., both input and output files are named on the command line. This style -is a little more foolproof, and it loses no functionality if you don't have -pipes. You can get this style on Unix too, if you prefer, by defining -TWO_FILE_COMMANDLINE in jconfig.h or in the Makefile. You MUST use this style -on any system that doesn't cope well with binary data fed through -stdin/stdout. - -Currently supported image file formats include raw-format PPM, raw-format PGM -(for monochrome images), and GIF. cjpeg recognizes the input image format -automatically, but you have to tell djpeg which format to generate. - -The only JPEG file format currently supported is a raw JPEG data stream. -Unless modified, the programs use the JFIF conventions for variables left -unspecified by the JPEG standard. (In particular, cjpeg generates a JFIF APP0 -marker.) Support for the JPEG-in-TIFF format will probably be added at some -future date. - -The command line switches for cjpeg are: - - -I Generate noninterleaved JPEG file (not yet supported). - - -Q quality Scale quantization tables to adjust quality. - Quality is 0 (worst) to 100 (best); default is 75. - (See below for more info.) - - -a Use arithmetic coding rather than Huffman coding. - (Not currently supported, see LEGAL ISSUES.) - - -o Perform optimization of entropy encoding parameters. - Without this, default Huffman or arithmetic - parameters are used. -o makes the JPEG file a tad - smaller, but compression uses much more memory. - Image quality is unaffected by -o. - - -d Enable debug printout. More -d's give more printout. - -Typically you'd use -Q settings of 50 or 75 or so. -Q 100 will generate a -quantization table of all 1's, meaning no quantization loss; then any -differences between input and output images are due to subsampling or to -roundoff error in the DCT or colorspace-conversion steps. -Q values below 50 -may be useful for making real small, low-quality images. Try -Q 2 (or so) for -some amusing Cubist effects. (Note that -Q values below about 25 generate -2-byte quantization tables, which are not decodable by pure baseline JPEG -decoders. cjpeg emits a warning message when you give such a -Q value.) +ARCHIVE LOCATIONS +================= -The command line switches for djpeg are: +The "official" archive site for this software is uunet.uu.net (Internet +address 137.39.1.2 or 192.48.96.2). The most recent released version can +always be found there in directory graphics/jpeg. This particular version +will be archived as jpegsrc.v2.tar.Z. If you are on the Internet, you can +retrieve files from UUNET by anonymous FTP. If you don't have FTP access, +UUNET's archives are also available via UUCP; contact postmaster@uunet.uu.net +for information on retrieving files that way. - -G Select GIF output format (implies -q, with default - of 256 colors). +Various other Internet sites maintain copies of the UUNET file, which may or +may not be up-to-date. In Europe, try nic.funet.fi (128.214.6.100; look in +directory pub/graphics/programs/jpeg). - -b Perform cross-block smoothing. This is quite - memory-intensive and only seems to improve the image - at very low quality settings (-Q 10 to 20 or so). +You can also obtain this software from CompuServe, in the GRAPHSUPPORT forum +(GO PICS), library 10; this version will be file jpsrc2.zip. - -g Force gray-scale output even if input is color. +If you are not reasonably handy at configuring and installing portable C +programs, you may have some difficulty installing this package. You may +prefer to obtain a pre-built executable version. A collection of pre-built +executables for various machines is currently available for anonymous FTP at +procyon.cis.ksu.edu (129.130.10.80 --- this number is due to change soon); +look under /pub/JPEG. The administrators of this system ask that FTP traffic +be limited to non-prime hours. For more information on this archive, please +contact Steve Davis (strat@cis.ksu.edu). This collection is not maintained by +the Independent JPEG Group, and programs in it may not be the latest version. - -q N Quantize to N colors. - -D Use Floyd-Steinberg dithering in color quantization. - - -2 Use two-pass color quantization (not yet supported). - - -d Enable debug printout. More -d's give more printout. - -Color quantization currently uses a rather shoddy algorithm (although it's not -so horrible when dithered). Because of this, the GIF output mode is not -recommended in the current release, except for gray-scale output. You can get -better results by applying ppmquant to the unquantized (PPM) output of djpeg, -then converting to GIF with ppmtogif. We expect to provide a considerably -better quantization algorithm in a future release. +SUPPORTING SOFTWARE +=================== -Note that djpeg *can* read noninterleaved JPEG files even though cjpeg can't -yet generate them. For most applications this is a nonissue, since hardly -anybody seems to be using noninterleaved format. +You will probably want Jef Poskanzer's PBMPLUS image software, which provides +many useful operations on PPM-format image files. In particular, it can +convert PPM images to and from a wide range of other formats. You can FTP +this free software from export.lcs.mit.edu (contrib/pbmplus*.tar.Z) or +ftp.ee.lbl.gov (pbmplus*.tar.Z). Unfortunately PBMPLUS is not nearly as +portable as the JPEG software is; you are likely to have difficulty making it +work on any non-Unix machine. -On a non-virtual-memory machine, you may run out of memory if you use -I or -o -in cjpeg, or -q ... -2 in djpeg, or try to read an interlaced GIF file. This -will be addressed eventually by replacing jvirtmem.c with something that uses -temporary files for large images (see TO DO). +If you are using X Windows you might want to use the xv or xloadimage viewers +to save yourself the trouble of converting PPM to some other format. Both of +these can be found in the contrib directory at export.lcs.mit.edu. +There will soon be a new release of xv that incorporates our software and thus +can read and write JPEG files directly. (NOTE: since xv internally reduces +all images to 8 bits/pixel, a JPEG file written by xv will not be very high +quality. Caveat user.) + + +SOFTWARE THAT'S NO HELP AT ALL +============================== + +Handmade Software's shareware PC program GIF2JPG produces files that are +totally incompatible with our programs. They use a proprietary format that is +an amalgam of GIF and JPEG representations. However, you can force GIF2JPG +to produce compatible files with its -j switch, and their decompression +program JPG2GIF can read our files (at least ones produced with our default +option settings). + +Unfortunately, most commercial JPEG implementations are also incompatible as +of this writing, especially programs released before summer 1991. The root of +the problem is that the ISO JPEG committee failed to specify a concrete file +format. Many vendors "filled in the blanks" on their own, creating +proprietary formats that no one else could read. (For example, none of the +early commercial JPEG implementations for the Macintosh were able to exchange +compressed files.) + +The file format we have adopted is called JFIF (see REFERENCES). This format +has been agreed to by a number of major commercial JPEG vendors, and we expect +that it will become the de facto standard. JFIF is a minimal representation; +work is also going forward to incorporate JPEG compression into the TIFF +standard, for use in "high end" applications that need to record a lot of +additional data about an image. We intend to support JPEG-in-TIFF in the +future. We hope that these two formats will be sufficient and that other, +incompatible JPEG file formats will not proliferate. + +Indeed, part of the reason for developing and releasing this free software is +to help force rapid convergence to de facto standards for JPEG file formats. +SUPPORT STANDARD, NON-PROPRIETARY FORMATS: demand JFIF or JPEG-in-TIFF! REFERENCES @@ -276,13 +183,25 @@ algorithm is Wallace's article in the April '91 CACM: Communications of the ACM, April 1991 (vol. 34 no. 4), pp. 30-44. (Adjacent articles in that issue discuss MPEG motion picture compression, applications of JPEG, and related topics.) We highly recommend reading that -article before looking at any of the JPEG software. +article before trying to understand the innards of any JPEG software. For more detail about the JPEG standard you pretty much have to go to the -draft standard, which is not nearly as intelligible as Wallace's article. -The current version is ISO/IEC Committee Draft CD 10918-1 dated 1991-03-15. -The standard is not presently available electronically; you must order a paper -copy through ISO. +draft standard (which is not nearly as intelligible as Wallace's article). +The standard is not now available electronically; you must order a paper copy +through ISO. In the US, copies may be ordered from ANSI Sales at (212) +642-4900. The standard is divided into two parts: Part 1 is the actual +specification, and Part 2 covers compliance testing methods. The current +"committee draft" version of Part 1 is titled "Digital Compression and Coding +of Continuous-tone Still Images, Part 1: Requirements and guidelines" and has +document number ISO/IEC CD 10918-1. (The alternate number SC2 N2215 should +also be mentioned when ordering.) This draft is expected to be superseded by +the Draft International Standard version around the end of November 1991. +Ordering info will be the same as above, but replace "CD" with "DIS" in the +document number (alternate number not yet known). The committee draft of +Part 2 is expected to be available around the end of December 1991. It will +be titled "Digital Compression and Coding of Continuous-tone Still Images, +Part 2: Compliance testing" and will have document number ISO/IEC CD 10918-2 +(alternate number not yet known). The JPEG standard does not specify all details of an interchangeable file format. For the omitted details we follow the "JFIF" conventions, revision @@ -292,28 +211,14 @@ format. For the omitted details we follow the "JFIF" conventions, revision 399A West Trimble Road San Jose, CA 95131 (408) 944-6300 -Requests can also be e-mailed to info@c3.pla.ca.us (this address good after -10/10/91). The same source can supply copies of the draft JPEG-in-TIFF specs. +Requests can also be e-mailed to info@c3.pla.ca.us. The same source can +supply copies of the draft JPEG-in-TIFF specs. If you want to understand this implementation, start by reading the "architecture" documentation file. Please read "codingrules" if you want to contribute any code. -SUPPORTING SOFTWARE -=================== - -You will probably want Jef Poskanzer's PBMPLUS image software; this provides -many useful operations on PPM-format image files. In particular, it can -convert PPM images to and from a wide range of other formats. You can FTP -this free software from export.lcs.mit.edu (contrib/pbmplus*.tar.Z) or -ftp.ee.lbl.gov (pbmplus*.tar.Z). - -If you are using X Windows you might want to use the xv or xloadimage viewers -to save yourself the trouble of converting PPM to some other format. -Both of these can be found in the contrib directory at export.lcs.mit.edu. - - LEGAL ISSUES ============ @@ -360,14 +265,21 @@ the foregoing paragraphs do. It appears that the arithmetic coding option of the JPEG spec is covered by -patents held by IBM, and possibly also patents of AT&T and Mitsubishi. Hence -arithmetic coding cannot legally be used without obtaining one or more -licenses. For this reason, support for arithmetic coding has been removed -from the free JPEG software. (Since arithmetic coding provides only a +patents owned by IBM and AT&T, as well as a pending Japanese patent of +Mitsubishi. Hence arithmetic coding cannot legally be used without obtaining +one or more licenses. For this reason, support for arithmetic coding has been +removed from the free JPEG software. (Since arithmetic coding provides only a marginal gain over the unpatented Huffman mode, it is unlikely that very many -people will choose to use it. If you do obtain such a license, contact -jpeg-info@uunet.uu.net for a copy of our arithmetic coding modules.) So far -as we are aware, there are no patent restrictions on the remaining code. +people will choose to use it. If you do obtain the necessary licenses, +contact jpeg-info@uunet.uu.net for a copy of our arithmetic coding modules.) +So far as we are aware, there are no patent restrictions on the remaining +code. + + +We are required to state that + "The Graphics Interchange Format(c) is the Copyright property of + CompuServe Incorporated. GIF(sm) is a Service Mark property of + CompuServe Incorporated." TO DO diff --git a/SETUP b/SETUP new file mode 100644 index 000000000..31f445c02 --- /dev/null +++ b/SETUP @@ -0,0 +1,235 @@ +SETUP instructions for the Independent JPEG Group's JPEG software +================================================================= + +This file explains how to configure and compile the JPEG software. We have +tried to make this software extremely portable and flexible, so that it can be +adapted to almost any environment. The downside of this decision is that the +installation process is not very automatic; you will need at least a little +familiarity with C programming and program build procedures for your system. + +This file contains general instructions, then sections of specific hints for +certain systems. You may save yourself considerable time if you scan the +whole file before starting to do anything. + +Before installing the software you must unpack the distributed source code. +Since you are reading this file, you have probably already succeeded in this +task. However, there is one potential trap if you are on a non-Unix system: +you may need to convert these files to the local standard text file format +(for example, if you are on MS-DOS you probably have to convert LF end-of-line +to CR/LF). If so, apply the conversion to all the files EXCEPT those whose +names begin with "test". The test files contain binary data; if you change +them in any way then the self-test will give bad results. + + +STEP 1: PREPARE A MAKEFILE +========================== + +First, select a makefile and copy it to "Makefile" (or whatever your version +of make uses as the default makefile name; for example, "makefile.mak" for +Borland C). We include several standard makefiles in the distribution: + + makefile.ansi: for Unix systems with ANSI-compatible C compilers. + makefile.unix: for Unix systems with non-ANSI C compilers. + makefile.mc5: for Microsoft C 5.x under MS-DOS. + makefile.mc6: for Microsoft C 6.x under MS-DOS. + makefile.tc: for Borland's Turbo C under MS-DOS. + makefile.pwc: for Mix Software's Power C under MS-DOS. + makefile.manx: for Manx Aztec C on Amigas. + makefile.sas: for SAS C on Amigas. + +If you don't see a makefile for your system, we recommend starting from either +makefile.ansi or makefile.unix, depending on whether your compiler accepts +ANSI C or not. Actually you should start with makefile.ansi whenever your +compiler supports ANSI-style function definitions; you don't need full ANSI +compatibility. The difference between the two makefiles is that makefile.unix +preprocesses the source code to convert function definitions to old-style C. +(Our thanks to Peter Deutsch of Aladdin Enterprises for the ansi2knr program.) + +If you don't know whether your compiler supports ANSI-style function +definitions, then take a look at config.c. It is a test program that will +help you figure out this fact, as well as some other facts you'll need in +later steps. You must compile and execute config.c by hand; the makefiles +don't provide any support for this. config.c may not compile the first try +(in fact, the whole idea is for it to fail if anything is going to). If you +get compile errors, fix them by editing config.c according to the directions +given in config.c. Once you get it to run, select a makefile according to the +advice it prints out, and make any other changes it recommends. + +Look over the selected Makefile and adjust options as needed. In particular +you may want to change the CC and CFLAGS definitions. For instance, if you +are using GCC, set CC=gcc. + +If you are on a system that doesn't use makefiles, you'll need to set up +project files (or whatever you do use) to compile all the source files and +link them into executable files cjpeg and djpeg. See the file lists in any of +the makefiles to find out which files go into each program (makcjpeg.lst and +makdjpeg.lst are handy summaries). + + +STEP 2: EDIT JCONFIG.H +====================== + +Look over jconfig.h and adjust #defines to reflect the properties of your +system and C compiler. (If you prefer, you can usually leave jconfig.h +unmodified and add -Dsymbol switches to the Makefile's CFLAGS definition.) + +If you have an ANSI-compliant C compiler, no changes should be necessary +except perhaps for RIGHT_SHIFT_IS_UNSIGNED and TWO_FILE_COMMANDLINE. For +older compilers other changes may be needed, depending on what ANSI features +are supported. + +If you don't know enough about C programming to understand the questions in +jconfig.h, then use config.c to figure out what to change. (See description +of config.c in step 1.) + +A note about TWO_FILE_COMMANDLINE: defining this selects the command line +syntax in which the input and output files are both named on the command line. +If it's not defined, the output image goes to standard output, and the input +can optionally come from standard input. You MUST use two-file style on any +system that doesn't cope well with binary data fed through stdin/stdout; this +is true for most MS-DOS compilers, for example. If you're not on a Unix +system, it's probably safest to assume you need two-file style. + + +STEP 3: MAKE +============ + +Now you should be able to "make" the software. + +If you have trouble with missing system include files or inclusion of the +wrong ones, look at jinclude.h (or use config.c, if you are not a C expert). + +If your compiler complains about big_sarray_control and big_barray_control +being undefined structures, you should be able to shut it up by adding +-DINCOMPLETE_TYPES_BROKEN to CFLAGS (or add #define INCOMPLETE_TYPES_BROKEN +to jconfig.h). + +There are a fair number of routines that do not use all of their parameters; +some compilers will issue warnings about this, which you can ignore. Any +other warning deserves investigation. + + +STEP 4: TEST +============ + +As a quick test of functionality we've included three small sample files: + testorig.jpg A reduced section of the well-known Lenna picture. + testimg.ppm The output of djpeg testorig.jpg + testimg.jpg The output of cjpeg testimg.ppm +(The two .jpg files aren't identical since JPEG is lossy.) If you can +generate duplicates of testimg.ppm and testimg.jpg then you probably have a +working port. + +With most of the makefiles, "make test" will perform the necessary +comparisons. If you're using a makefile that doesn't provide this option, run +djpeg and cjpeg to generate testout.ppm and testout.jpg, then compare these to +testimg.* with whatever file comparison tool you have. The files should be +bit-for-bit identical. + +NOTE: this is far from an exhaustive test of the JPEG software; some modules, +such as color quantization and GIF I/O, are not exercised at all. It's just a +quick test to give you some confidence that you haven't missed something +major. + +If the test passes, you can copy the executable files cjpeg and djpeg to +wherever you normally install programs. Read the file USAGE to learn more +about using the programs. + + +OPTIONAL STUFF +============== + +We distribute the software with support for RLE image files (Utah Raster +Toolkit format) disabled, because the RLE support won't compile without the +Utah library. If you have URT version 3.0, you can enable RLE support as +follows: + 1. #define RLE_SUPPORTED in jconfig.h or in the Makefile. + 2. Add a -I option to CFLAGS in the Makefile for the directory + containing the URT .h files (typically the "include" + subdirectory of the URT distribution). + 3. Add -L... -lrle to LDLIBS in the Makefile, where ... specifies + the directory containing the URT "librle.a" file (typically the + "lib" subdirectory of the URT distribution). + +If you want to incorporate the JPEG code as subroutines in a larger program, +we recommend that you make libjpeg.a. Then use the jconfig.h and jpegdata.h +files as your interface to the JPEG functions, and link libjpeg.a with your +program. Your surrounding program will have to provide functionality similar +to what's in jcmain.c or jdmain.c, and you may want to replace jerror.c and +possibly other modules depending on your needs. See the "architecture" file +for more info. If it seems to you that the system structure doesn't +accommodate what you want to do, please contact the authors. + +CAUTION: When you use the JPEG code as subroutines, we recommend that you make +any required configuration changes by modifying jconfig.h, not by adding -D +switches to the Makefile. Otherwise you must be sure to provide the same -D +switches when compiling any program that includes the JPEG .h files. + +If you need to make a smaller version of the JPEG software, some optional +functions can be removed at compile time. See the xxx_SUPPORTED #defines in +jconfig.h. If at all possible, we recommend that you leave in decoder support +for all valid JPEG files, to ensure that you can read anyone's output. +Restricting your encoder, or removing optional functions like block smoothing, +won't hurt compatibility. Taking out support for image file formats that you +don't use is the most painless way to make the programs smaller. + + +NOTES FOR SPECIFIC SYSTEMS +========================== + +We welcome reports on changes needed for systems not mentioned here. +Submit 'em to jpeg-info@uunet.uu.net. Also, config.c is fairly new and not +yet thoroughly tested; if it's wrong about how to configure the JPEG software +for your system, please let us know. + + +HP/Apollo DOMAIN: + +At least in version 10.3.5, the C compiler is ANSI but the system include +files are not. Use makefile.ansi and add -DNONANSI_INCLUDES to CFLAGS. + +HP-UX: + +If you have HP-UX 7.05 or later with the "software development" C compiler, +then you can use makefile.ansi. Add "-Aa" to the CFLAGS line in the +makefile. If you have a pre-7.05 system, or if you are using the non-ANSI C +compiler delivered with a minimum HP-UX 8.0 system, then you must use +makefile.unix (and do NOT add -Aa). Also, adding "-lmalloc" to LDLIBS is +recommended if you have libmalloc.a (it seems not to be present in minimum +8.0). + +On HP series 800 machines, the HP C compiler is buggy in revisions prior to +A.08.07. If you get complaints about "not a typedef name", you'll have to +convert the code to K&R style (i.e., use makefile.unix). + +IBM RS/6000 AIX: + +The CFLAGS switch to make the compiler define __STDC__ is "-qlanglvl=ansi". + +Macintosh Think C: + +You'll have to prepare project files for cjpeg and djpeg; we don't include +those in the distribution since they are not text files. The COBJECTS and +DOBJECTS lists in makefile.unix show which files should be included in each +project. Also add the ANSI and Unix C libraries in a separate segment. You +may need to divide the JPEG files into more than one segment; you can do this +pretty much as you please. + +If you have Think C version 5.0 you should be able to just turn on __STDC__ +through the compiler switch that enables that. With version 4.0 you must +manually edit jconfig.h. (You can #define __STDC__, but also #define const.) + +Microsoft C for MS-DOS: + +Some versions of MS C fail with an "out of macro expansion space" error +because they can't cope with the macro TRACEMS8 (defined in jpegdata.h). +If this happens to you, the easiest solution is to change TRACEMS8 to +expand to nothing. You'll lose the ability to dump out JPEG coefficient +tables with djpeg -d -d, but at least you can compile. + +Sun: + +Don't forget to add -DBSD to CFLAGS. If you are using GCC on SunOS 4.0.1 or +earlier, you will need to add -DNONANSI_INCLUDES to CFLAGS (your compiler may +be ANSI, but your system include files aren't). I've gotten conflicting +reports on whether this is still necessary on SunOS 4.1 or later. diff --git a/USAGE b/USAGE new file mode 100644 index 000000000..409cb4152 --- /dev/null +++ b/USAGE @@ -0,0 +1,162 @@ +USAGE instructions for the Independent JPEG Group's JPEG software +================================================================= + +This distribution contains software to implement JPEG image compression and +decompression. JPEG (pronounced "jay-peg") is a standardized compression +method for full-color and gray-scale images. JPEG is intended for +"real-world" scenes; cartoons and other non-realistic images are not its +strong suit. JPEG is lossy, meaning that the output image is not necessarily +identical to the input image. Hence you should not use JPEG if you have to +have identical output bits. However, on typical images of real-world scenes, +very good compression levels can be obtained with no visible change, and +amazingly high compression levels can be obtained if you can tolerate a +low-quality image. + +This file describes usage of the standard programs "cjpeg" and "djpeg" that +can be built directly from the distributed software. See the README file for +hints on incorporating the JPEG software into other programs. + +If you are on a Unix machine you may prefer to read the Unix-style manual +pages in files cjpeg.1 and djpeg.1. + +NOTE: at some point we will probably redesign the user interface, so the +command line switches described here will change. + + +We provide two programs, cjpeg to compress an image file into JPEG format, +and djpeg to decompress a JPEG file back into a conventional image format. + +On Unix-like systems, you say: + cjpeg [switches] [imagefile] >jpegfile +or + djpeg [switches] [jpegfile] >imagefile +The programs read the specified input file, or standard input if none is +named. They always write to standard output (with trace/error messages to +standard error). These conventions are handy for piping images between +programs. + +On PC, Macintosh, and Amiga systems, you say: + cjpeg [switches] imagefile jpegfile +or + djpeg [switches] jpegfile imagefile +i.e., both input and output files are named on the command line. This style +is a little more foolproof, and it loses no functionality if you don't have +pipes. (You can get this style on Unix too, if you prefer, by defining +TWO_FILE_COMMANDLINE; see SETUP.) + +The currently supported image file formats are: PPM (PBMPLUS color format), +PGM (PBMPLUS gray-scale format), GIF, Targa, and RLE (Utah Raster Toolkit +format). (RLE is supported only if the URT library is available.) +cjpeg recognizes the input image format automatically, with the exception +of some Targa-format files. + +The only JPEG file format currently supported is the JFIF format. Support for +the TIFF/JPEG format will probably be added at some future date. + + +The command line switches for cjpeg are: + + -Q quality Scale quantization tables to adjust image quality. + Quality is 0 (worst) to 100 (best); default is 75. + (See below for more info.) + + -o Perform optimization of entropy encoding parameters. + Without this, default encoding parameters are used. + -o usually makes the JPEG file a little smaller, but + cjpeg runs much slower. Image quality and speed of + decompression are unaffected by -o. + + -T Input file is Targa format. Targa files that contain + an "identification" field will not be automatically + recognized by cjpeg; for such files you must specify + -T to force cjpeg to treat the input as Targa format. + + -I Generate noninterleaved JPEG file (not yet supported). + + -a Use arithmetic coding rather than Huffman coding. + (Not currently supported for legal reasons.) + + -d Enable debug printout. More -d's give more printout. + Also, version information is printed at startup. + +The -Q switch lets you trade off compressed file size against quality of the +reconstructed image: the higher the -Q setting, the larger the JPEG file, and +the closer the output image will be to the original input. Normally you want +to use the lowest -Q setting (smallest file) that decompresses into something +visually indistinguishable from the original image. For this purpose the -Q +setting should be between 50 and 95; the default of 75 is often about right. +If you see defects at -Q 75, then go up 5 or 10 counts at a time until you are +happy with the output image. (The optimal setting will vary from one image to +another.) + +-Q 100 will generate a quantization table of all 1's, eliminating loss in the +quantization step (but there is still information loss in subsampling, as well +as roundoff error). This setting is mainly of interest for experimental +purposes. -Q values above about 95 are NOT recommended for normal use; the +compressed file size goes up dramatically for hardly any gain in output image +quality. + +In the other direction, -Q values below 50 will produce very small files of +low image quality. Settings around 5 to 10 might be useful in preparing an +index of a large image library, for example. Try -Q 2 (or so) for some +amusing Cubist effects. (Note: -Q values below about 25 generate 2-byte +quantization tables, which are considered optional in the JPEG standard. +cjpeg emits a warning message when you give such a -Q value, because some +commercial JPEG programs may be unable to decode the resulting file.) + + +The command line switches for djpeg are: + + -G Select GIF output format (implies -q, with default + of 256 colors). + + -P Select PPM or PGM output format (this is the default). + PGM is emitted if the JPEG file is gray-scale or if -g + is specified. + + -R Select RLE output format. Requires URT library. + + -T Select Targa output format. Gray-scale format is + emitted if the JPEG file is gray-scale or if -g is + specified; otherwise, colormapped format is emitted + if -q is specified; otherwise, 24-bit full-color + format is emitted. + + -b Perform cross-block smoothing. This is quite + memory-intensive and only seems to improve the image + at very low quality settings (-Q 10 to 20 or so). + At normal -Q settings it may make the image worse. + + -g Force gray-scale output even if input is color. + + -q N Quantize to N colors. + + -D Do NOT use dithering in color quantization. + By default, Floyd-Steinberg dithering is applied when + quantizing colors, but on some images dithering may + result in objectionable "graininess". If that + happens, you can turn off dithering with -D. + + -2 Use two-pass color quantization (not yet supported). + + -d Enable debug printout. More -d's give more printout. + Also, version information is printed at startup. + +Color quantization currently uses a rather shoddy algorithm (although it's not +as horrible when dithered). Because of this, the GIF output mode is NOT +RECOMMENDED in the current release, except for gray-scale output. You can get +better results by applying ppmquant to the unquantized (PPM) output of djpeg, +then converting to GIF with ppmtogif. (See SUPPORTING SOFTWARE in the README +file.) We expect to provide a considerably better quantization algorithm in a +future release. (The same applies to colormapped RLE or Targa output, of +course.) + +Note that djpeg *can* read noninterleaved JPEG files even though cjpeg can't +yet generate them. For most applications this is a nonissue, since hardly +anybody seems to be using noninterleaved format. + +On a non-virtual-memory machine, you may run out of memory if you use -I or -o +in cjpeg, or -q ... -2 in djpeg, or try to read an interlaced GIF file, or try +to read or write an RLE file, or try to read an interlaced or bottom-up Targa +file. This will be addressed soon by replacing jvirtmem.c with something that +uses temporary files for large images. diff --git a/ansi2knr.c b/ansi2knr.c index 77d3d8197..04e4789eb 100644 --- a/ansi2knr.c +++ b/ansi2knr.c @@ -1,6 +1,7 @@ /* * Received from Peter Deutsch (ghost@aladdin.com) * Fri, 26 Apr 91 10:10:10 PDT + * Small portability improvements by Tom Lane */ /* Copyright (C) 1989, 1991 Aladdin Enterprises. All rights reserved. @@ -174,34 +175,32 @@ BY ANY OTHER PARTY. /* ansi2knr.c */ /* Convert ANSI function declarations to K&R syntax */ + #include #include #ifdef BSD -# include -# define strchr index +#include +#define strchr index #else -# ifdef VMS +#ifdef VMS extern char *strcat(), *strchr(), *strcpy(), *strupr(); extern int strcmp(), strlen(), strncmp(); -# else -# include -# endif +#else +#include +#endif #endif #ifdef MSDOS -# include +#include #else -# ifdef VMS +#ifdef VMS extern char *malloc(); extern void free(); -# else -# ifdef BSD - extern char *malloc(); -# else -# include -# endif -# endif +#else + extern char *malloc(); + extern int free(); +#endif #endif /* Usage: @@ -226,12 +225,13 @@ BY ANY OTHER PARTY. #define isidchar(ch) (isalnum(ch) || (ch) == '_') #define isidfirstchar(ch) (isalpha(ch) || (ch) == '_') +int main(argc, argv) int argc; char *argv[]; { FILE *in, *out; #define bufsize 500 /* arbitrary size */ - char buf[bufsize]; + char buf[bufsize+1]; char *line; switch ( argc ) { @@ -296,12 +296,13 @@ skipspace(p, dir) /* * Write blanks over part of a string. */ -void +int writeblanks(start, end) char *start; char *end; { char *p; for ( p = start; p < end; p++ ) *p = ' '; + return 0; } /* @@ -313,6 +314,7 @@ writeblanks(start, end) * -1 - may be the beginning of a function definition, * append another line and look again. */ +int test1(buf) char *buf; { register char *p = buf; @@ -359,6 +361,7 @@ test1(buf) return contin; } +int convert1(buf, out) char *buf; FILE *out; diff --git a/cjpeg.1 b/cjpeg.1 new file mode 100644 index 000000000..9a1832204 --- /dev/null +++ b/cjpeg.1 @@ -0,0 +1,130 @@ +.TH CJPEG 1 "11 December 1991" +.SH NAME +cjpeg \- compress an image file to a JPEG file +.SH SYNOPSIS +.B cjpeg +[ +.BI \-Q " quality" +] +[ +.B \-oTIad +] +[ +.I filename +] +.LP +.SH DESCRIPTION +.LP +.B cjpeg +compresses the named image file, or the standard input if no file is +named, and produces a JPEG/JFIF file on the standard output. +The currently supported image file formats are: PPM (PBMPLUS color +format), PGM (PBMPLUS gray-scale format), GIF, Targa, and RLE (Utah Raster +Toolkit format). (RLE is supported only if the URT library is available.) +.SH OPTIONS +.TP +.BI \-Q " quality" +Scale quantization tables to adjust image quality. Quality is 0 (worst) to +100 (best); default is 75. (See below for more info.) +.TP +.B \-o +Perform optimization of entropy encoding parameters. Without this, default +encoding parameters are used. +.B \-o +usually makes the JPEG file a little smaller, but +.B cjpeg +runs much slower. Image quality and speed of decompression are unaffected by +.BR \-o . +.TP +.B \-T +Input file is Targa format. Targa files that contain an "identification" +field will not be automatically recognized by +.BR cjpeg ; +for such files you must specify +.B \-T +to force +.B cjpeg +to treat the input as Targa format. +.TP +.B \-I +Generate noninterleaved JPEG file (not yet supported). +.TP +.B \-a +Use arithmetic coding rather than Huffman coding (not currently +supported for legal reasons). +.TP +.B \-d +Enable debug printout. More +.BR \-d 's +give more output. Also, version information is printed at startup. +.PP +The +.B \-Q +switch lets you trade off compressed file size against quality of the +reconstructed image: the higher the +.B \-Q +setting, the larger the JPEG file, and the closer the output image will be to +the original input. Normally you want to use the lowest +.B \-Q +setting (smallest file) that decompresses into something visually +indistinguishable from the original image. For this purpose the +.B \-Q +setting should be between 50 and 95; the default of 75 is often about right. +If you see defects at +.B \-Q +75, then go up 5 or 10 counts at a time until you are happy with the output +image. (The optimal setting will vary from one image to another.) +.PP +.B \-Q +100 will generate a quantization table of all 1's, eliminating loss in the +quantization step (but there is still information loss in subsampling, as well +as roundoff error). This setting is mainly of interest for experimental +purposes. +.B \-Q +values above about 95 are +.B not +recommended for normal use; the compressed file size goes up dramatically for +hardly any gain in output image quality. +.PP +In the other direction, +.B \-Q +values below 50 will produce very small files of low image quality. Settings +around 5 to 10 might be useful in preparing an index of a large image library, +for example. Try +.B \-Q +2 (or so) for some amusing Cubist effects. (Note: +.B \-Q +values below about 25 generate 2-byte quantization tables, which are +considered optional in the JPEG standard. +.B cjpeg +emits a warning message when you give such a +.B \-Q +value, because some commercial JPEG programs may be unable to decode the +resulting file.) +.SH EXAMPLES +.LP +This example compresses the PPM file foo.ppm with a quality factor of +60 and saves the output as foo.jpg: +.IP +.B cjpeg \-Q +.I 60 foo.ppm +.B > +.I foo.jpg +.SH SEE ALSO +.BR djpeg (1) +.br +Wallace, Gregory K. "The JPEG Still Picture Compression Standard", +Communications of the ACM, April 1991 (vol. 34, no. 4), pp. 30-44. +.SH AUTHOR +Independent JPEG Group +.SH BUGS +Arithmetic coding and interleaved output not yet supported. +.PP +Not all variants of Targa file format are supported. +.PP +The +.B -T +switch is not a bug, it's a feature. (It would be a bug if the Targa format +designers had not been clueless.) +.PP +Not as fast as we'd like. diff --git a/config.c b/config.c new file mode 100644 index 000000000..938fd1970 --- /dev/null +++ b/config.c @@ -0,0 +1,403 @@ +/* + * config.c + * + * Copyright (C) 1991, Thomas G. Lane. + * This file is part of the Independent JPEG Group's software. + * For conditions of distribution and use, see the accompanying README file. + */ + +/* + * This program is intended to help you determine how to configure the JPEG + * software for installation on a particular system. The idea is to try to + * compile and execute this program. If your compiler fails to compile the + * program, make changes as indicated in the comments below. Once you can + * compile the program, run it, and it will tell you how to set the various + * switches in jconfig.h and in your Makefile. + * + * This could all be done automatically if we could assume we were on a Unix + * system, but we don't want to assume that, so you'll have to edit and + * recompile this program until it works. + * + * As a general rule, each time you try to compile this program, + * pay attention only to the *first* error message you get from the compiler. + * Many C compilers will issue lots of spurious error messages once they + * have gotten confused. Go to the line indicated in the first error message, + * and read the comments preceding that line to see what to change. + * + * Almost all of the edits you may need to make to this program consist of + * changing a line that reads "#define SOME_SYMBOL" to "#undef SOME_SYMBOL", + * or vice versa. This is called defining or undefining that symbol. + */ + + +/* First we must see if your system has the include files we need. + * We start out with the assumption that your system follows the ANSI + * conventions for include files. If you get any error in the next dozen + * lines, undefine INCLUDES_ARE_ANSI. + */ + +#define INCLUDES_ARE_ANSI /* replace 'define' by 'undef' if error here */ + +#ifdef INCLUDES_ARE_ANSI /* this will be skipped if you undef... */ +#include /* If you ain't got this, you ain't got C. */ +#ifdef __SASC /* Amiga SAS C provides size_t in stddef.h. */ +#include /* (They are wrong...) */ +#endif +#include /* size_t might be here too. */ +typedef size_t my_size_t; /* The payoff: do we have size_t now? */ +#include /* Check other ANSI includes we use. */ +#endif + + +/* If your system doesn't follow the ANSI conventions, we have to figure out + * what it does follow. If you didn't get an error before this line, you can + * ignore everything down to "#define HAVE_ANSI_DEFINITIONS". + */ + +#ifndef INCLUDES_ARE_ANSI /* skip these tests if INCLUDES_ARE_ANSI */ + +#include /* If you ain't got this, you ain't got C. */ + +/* jinclude.h will try to include if you don't set + * INCLUDES_ARE_ANSI. We need to test whether that include file is provided. + * If you get an error here, undefine HAVE_TYPES_H. + */ + +#define HAVE_TYPES_H + +#ifdef HAVE_TYPES_H +#include +#endif + +/* We have to see if your string functions are defined by + * strings.h (BSD convention) or string.h (everybody else). + * We try the non-BSD convention first; define BSD if the compiler + * says it can't find string.h. + */ + +#undef BSD + +#ifdef BSD +#include +#else +#include +#endif + +/* Usually size_t is defined in stdio.h, sys/types.h, and/or string.h. + * If not, you'll get an error on the "typedef size_t my_size_t;" line below. + * In that case, you'll have to search through your system library to + * figure out which include file defines "size_t". Look for a line that + * says "typedef something-or-other size_t;" (stddef.h and stdlib.h are + * good places to look first). Then, change the line below that says + * "#include " to instead include the file + * you found size_t in, and define NEED_SPECIAL_INCLUDE. + */ + +#undef NEED_SPECIAL_INCLUDE /* assume we DON'T need it, for starters */ + +#ifdef NEED_SPECIAL_INCLUDE +#include +#endif + +typedef size_t my_size_t; /* The payoff: do we have size_t now? */ + + +#endif /* INCLUDES_ARE_ANSI */ + + + +/* The next question is whether your compiler supports ANSI-style function + * definitions. You need to know this in order to choose between using + * makefile.ansi and using makefile.unix. + * The #define line below is set to assume you have ANSI function definitions. + * If you get an error in this group of lines, undefine HAVE_ANSI_DEFINITIONS. + */ + +#define HAVE_ANSI_DEFINITIONS + +#ifdef HAVE_ANSI_DEFINITIONS +int testfunction (int arg1, int * arg2); /* check prototypes */ + +struct methods_struct { /* check method-pointer declarations */ + int (*error_exit) (char *msgtext); + int (*trace_message) (char *msgtext); +}; + +int testfunction (int arg1, int * arg2) /* check definitions */ +{ + return arg2[arg1]; +} +#endif + + +/* Now we want to find out if your compiler knows what "unsigned char" means. + * If you get an error on the "unsigned char un_char;" line, + * then undefine HAVE_UNSIGNED_CHAR. + */ + +#define HAVE_UNSIGNED_CHAR + +#ifdef HAVE_UNSIGNED_CHAR +unsigned char un_char; +#endif + + +/* Now we want to find out if your compiler knows what "unsigned short" means. + * If you get an error on the "unsigned short un_short;" line, + * then undefine HAVE_UNSIGNED_SHORT. + */ + +#define HAVE_UNSIGNED_SHORT + +#ifdef HAVE_UNSIGNED_SHORT +unsigned short un_short; +#endif + + +/* Now we want to find out if your compiler understands type "void". + * If you get an error anywhere in here, undefine HAVE_VOID. + */ + +#define HAVE_VOID + +#ifdef HAVE_VOID +typedef void * void_ptr; /* check void * */ +typedef void (*void_func) (); /* check ptr to function returning void */ + +void testfunction2 (arg1, arg2) /* check void function result */ + void_ptr arg1; + void_func arg2; +{ + char * locptr = (char *) arg1; /* check casting to and from void * */ + arg1 = (void *) locptr; + (*arg2) (1, 2); /* check call of fcn returning void */ +} +#endif + + +/* Now we want to find out if your compiler knows what "const" means. + * If you get an error here, undefine HAVE_CONST. + */ + +#define HAVE_CONST + +#ifdef HAVE_CONST +static const int carray[3] = {1, 2, 3}; + +int testfunction3 (arg1) + const int arg1; +{ + return carray[arg1]; +} +#endif + + + +/************************************************************************ + * OK, that's it. You should not have to change anything beyond this + * point in order to compile and execute this program. (You might get + * some warnings, but you can ignore them.) + * When you run the program, it will make a couple more tests that it + * can do automatically, and then it will print out a summary of the changes + * that you need to make to the makefile and jconfig.h. + ************************************************************************ + */ + + +static int any_changes = 0; + +int new_change () +{ + if (! any_changes) { + printf("\nMost of the changes recommended by this program can be made either\n"); + printf("by editing jconfig.h, or by adding -Dsymbol switches to the CFLAGS\n"); + printf("line in your Makefile. (Some PC compilers expect /Dsymbol instead.)\n"); + printf("The CFLAGS method is simpler, but if your system doesn't use makefiles,\n"); + printf("or if your compiler doesn't support -D, then you must change jconfig.h.\n"); + any_changes = 1; + } + printf("\n"); /* blank line before each problem report */ + return 0; +} + + +int test_char_sign (arg) + int arg; +{ + if (arg == 189) { /* expected result for unsigned char */ + new_change(); + printf("You should add -DCHAR_IS_UNSIGNED to CFLAGS,\n"); + printf("or else remove the /* */ comment marks from the line\n"); + printf("/* #define CHAR_IS_UNSIGNED */ in jconfig.h.\n"); + printf("(Be sure to delete the space before the # character too.)\n"); + } + else if (arg != -67) { /* expected result for signed char */ + new_change(); + printf("Hmm, it seems 'char' is less than eight bits wide on your machine.\n"); + printf("I fear the JPEG software will not work at all.\n"); + } + return 0; +} + + +int test_shifting (arg) + long arg; +/* See whether right-shift on a long is signed or not. */ +{ + long res = arg >> 4; + + if (res == 0x80817F4L) { /* expected result for unsigned */ + new_change(); + printf("You must add -DRIGHT_SHIFT_IS_UNSIGNED to CFLAGS,\n"); + printf("or else remove the /* */ comment marks from the line\n"); + printf("/* #define RIGHT_SHIFT_IS_UNSIGNED */ in jconfig.h.\n"); + } + else if (res != -0x7F7E80CL) { /* expected result for signed */ + new_change(); + printf("Right shift isn't acting as I expect it to.\n"); + printf("I fear the JPEG software will not work at all.\n"); + } + return 0; +} + + +int main (argc, argv) + int argc; + char ** argv; +{ + char signed_char_check = (char) (-67); + + printf("Results of configuration check for Independent JPEG Group's software:\n"); + printf("\nIf there's not a specific makefile provided for your compiler,\n"); +#ifdef HAVE_ANSI_DEFINITIONS + printf("you should use makefile.ansi as the starting point for your Makefile.\n"); +#else + printf("you should use makefile.unix as the starting point for your Makefile.\n"); +#endif + + /* Check whether we have all the ANSI features, */ + /* and whether this agrees with __STDC__ being predefined. */ +#ifdef __STDC__ +#define MY__STDC__ /* ANSI compilers won't allow redefining __STDC__ */ +#endif + +#ifdef HAVE_ANSI_DEFINITIONS +#ifdef HAVE_UNSIGNED_CHAR +#ifdef HAVE_UNSIGNED_SHORT +#ifdef HAVE_CONST +#define HAVE_ALL_ANSI_FEATURES +#endif +#endif +#endif +#endif + +#ifdef HAVE_ALL_ANSI_FEATURES +#ifndef MY__STDC__ + new_change(); + printf("Your compiler doesn't claim to be ANSI-compliant, but it is close enough\n"); + printf("for me. Either add -D__STDC__ to CFLAGS, or add #define __STDC__ at the\n"); + printf("beginning of jinclude.h (NOT jconfig.h).\n"); + printf("Some compilers will not let you do this: they will complain that __STDC__\n"); + printf("is a reserved name. In that case you have a compiler that really is ANSI,\n"); + printf("but you have to give it a special switch (often -ansi) to make it so.\n"); + printf("Check your compiler documentation and add the proper switch to CFLAGS.\n"); +#define MY__STDC__ +#endif +#else /* !HAVE_ALL_ANSI_FEATURES */ +#ifdef MY__STDC__ + new_change(); + printf("Your compiler claims to be ANSI-compliant, but it is lying!\n"); + printf("Either add -U__STDC__ to CFLAGS, or add #undef __STDC__\n"); + printf("at the beginning of jinclude.h (NOT jconfig.h).\n"); +#undef MY__STDC__ +#endif +#endif /* HAVE_ALL_ANSI_FEATURES */ + +#ifndef MY__STDC__ + +#ifdef HAVE_ANSI_DEFINITIONS + new_change(); + printf("You should add -DPROTO to CFLAGS, or else take out the several\n"); + printf("#ifdef/#else/#endif lines surrounding #define PROTO in jconfig.h.\n"); + printf("(Leave only one #define PROTO line.)\n"); +#endif + +#ifdef HAVE_UNSIGNED_CHAR +#ifdef HAVE_UNSIGNED_SHORT + new_change(); + printf("You should add -DHAVE_UNSIGNED_CHAR and -DHAVE_UNSIGNED_SHORT\n"); + printf("to CFLAGS, or else take out the #ifdef __STDC__/#endif lines\n"); + printf("surrounding #define HAVE_UNSIGNED_CHAR and #define HAVE_UNSIGNED_SHORT\n"); + printf("in jconfig.h.\n"); +#else /* only unsigned char */ + new_change(); + printf("You should add -DHAVE_UNSIGNED_CHAR to CFLAGS,\n"); + printf("or else move #define HAVE_UNSIGNED_CHAR outside the\n"); + printf("#ifdef __STDC__/#endif lines surrounding it in jconfig.h.\n"); +#endif +#else /* !HAVE_UNSIGNED_CHAR */ +#ifdef HAVE_UNSIGNED_SHORT + new_change(); + printf("You should add -DHAVE_UNSIGNED_SHORT to CFLAGS,\n"); + printf("or else move #define HAVE_UNSIGNED_SHORT outside the\n"); + printf("#ifdef __STDC__/#endif lines surrounding it in jconfig.h.\n"); +#endif +#endif /* HAVE_UNSIGNED_CHAR */ + +#ifdef HAVE_CONST + new_change(); + printf("You can delete the #define const line from jconfig.h.\n"); + printf("(But things should still work if you don't.)\n"); +#endif + +#endif /* MY__STDC__ */ + + test_char_sign((int) signed_char_check); + + test_shifting(-0x7F7E80B1L); + +#ifndef HAVE_VOID + new_change(); + printf("You should add -Dvoid=char to CFLAGS,\n"); + printf("or else remove the /* */ comment marks from the line\n"); + printf("/* #define void char */ in jconfig.h.\n"); + printf("(Be sure to delete the space before the # character too.)\n"); +#endif + +#ifdef INCLUDES_ARE_ANSI +#ifndef MY__STDC__ + new_change(); + printf("You should add -DINCLUDES_ARE_ANSI to CFLAGS, or else add\n"); + printf("#define INCLUDES_ARE_ANSI at the beginning of jinclude.h (NOT jconfig.h).\n"); +#endif +#else /* !INCLUDES_ARE_ANSI */ +#ifdef MY__STDC__ + new_change(); + printf("You should add -DNONANSI_INCLUDES to CFLAGS, or else add\n"); + printf("#define NONANSI_INCLUDES at the beginning of jinclude.h (NOT jconfig.h).\n"); +#endif +#ifdef NEED_SPECIAL_INCLUDE + new_change(); + printf("In jinclude.h, change the line reading #include \n"); + printf("to instead include the file you found size_t in.\n"); +#else /* !NEED_SPECIAL_INCLUDE */ +#ifndef HAVE_TYPES_H + new_change(); + printf("In jinclude.h, delete the line reading #include .\n"); +#endif +#endif /* NEED_SPECIAL_INCLUDE */ +#ifdef BSD + new_change(); + printf("You should add -DBSD to CFLAGS, or else add\n"); + printf("#define BSD at the beginning of jinclude.h (NOT jconfig.h).\n"); +#endif +#endif /* INCLUDES_ARE_ANSI */ + + if (any_changes) { + printf("\nI think that's everything...\n"); + } else { + printf("\nI think jconfig.h is OK as distributed.\n"); + } + + return any_changes; +} diff --git a/djpeg.1 b/djpeg.1 new file mode 100644 index 000000000..5eec670a8 --- /dev/null +++ b/djpeg.1 @@ -0,0 +1,114 @@ +.TH DJPEG 1 "11 December 1991" +.SH NAME +djpeg \- decompress a JPEG file to an image file +.SH SYNOPSIS +.B djpeg +[ +.B \-GPRTbgD2d +] +[ +.BI \-q " N" +] +[ +.I filename +] +.LP +.SH DESCRIPTION +.LP +.B djpeg +decompresses the named JPEG file, or the standard input if no file is named, +and produces an image file on the standard output. PPM, GIF, Targa, or RLE +output format can be selected. (RLE is supported only if the URT library is +available.) +.LP +The color quantization algorithm is currently shoddy. Because of this, the +GIF output mode is not recommended in the current release, except for +gray-scale output (obtained with +.BR \-g ). +.SH OPTIONS +.TP +.B \-G +Select GIF output format (implies +.BR \-q , +with default of 256 colors). +Currently the color quantization uses a shoddy algorithm and external +quantization (e.g. +.IR ppmquant , +.IR rlequant ) +is recommended before conversion to GIF format. +.TP +.B \-P +Select PPM or PGM output format (this is the default). PGM is emitted if the +JPEG file is gray-scale or if +.B \-g +is specified. +.TP +.B \-R +Select RLE output format. Requires URT library. +.TP +.B \-T +Select Targa output format. Gray-scale format is emitted if the JPEG file is +gray-scale or if +.B \-g +is specified; otherwise, colormapped format is emitted if +.B \-q +is specified; otherwise, 24-bit full-color format is emitted. +.TP +.B \-b +Perform cross-block smoothing. This is quite memory-intensive and only seems +to improve the image at low quality settings (\fB\-Q\fR 10 to 20 or so). +At normal +.B \-Q +settings it may make the image worse. +.TP +.B \-g +Force gray-scale output even if input is color. +.TP +.BI \-q " N" +Quantize to N colors. +.TP +.B \-D +Do not use dithering in color quantization. By default, Floyd-Steinberg +dithering is applied when quantizing colors, but on some images dithering may +result in objectionable "graininess". If that happens, you can turn off +dithering with +.BR \-D . +.TP +.B \-2 +Use two-pass color quantization (not yet supported). +.TP +.B \-d +Enable debug printout. More +.BR \-d 's +give more output. Also, version information is printed at startup. +.SH EXAMPLES +.LP +This example decompresses the JPEG file foo.jpg and saves the output +as a gray-scale image in foo.pgm: +.IP +.B djpeg \-g +.I foo.jpg +.B > +.I foo.pgm +.SH SEE ALSO +.BR cjpeg (1) +.br +.BR ppmquant (1) +[From the PBMplus distribution] +.br +.BR rlequant (1) +[From the Utah Raster Toolkit distribution] +.br +Wallace, Gregory K. "The JPEG Still Picture Compression Standard", +Communications of the ACM, April 1991 (vol. 34, no. 4), pp. 30-44. +.SH AUTHOR +Independent JPEG Group +.SH BUGS +.B djpeg +currently uses a shoddy color quantization algorithm. This leads to +poor GIF file output. Two-pass color quantization is not yet +supported. +.PP +Arithmetic coding is not supported for legal reasons. +.PP +Not as fast as we'd like. diff --git a/egetopt.c b/egetopt.c index 8183d358b..dbfd7fd55 100644 --- a/egetopt.c +++ b/egetopt.c @@ -40,21 +40,27 @@ * UUCP: ...!ames!fxgrp!ljz * * May, 1988 + * + * Modified for use in free JPEG code: + * + * Ed Hanway + * UUCP: uunet!sisd!jeh + * + * October, 1991 */ -/* - * If you want, include stdio.h or something where EOF and NULL are defined. - * However, egetopt() is written so as not to need stdio.h, which should - * make it significantly smaller on some systems. +/* The original egetopt.c was written not to need stdio.h. + * For the JPEG code this is an unnecessary and unportable assumption. + * Also, we make all the variables and routines "static" to avoid + * possible conflicts with a system-library version of getopt. + * + * In the JPEG code, this file is compiled by #including it in jcmain.c + * or jdmain.c. Since ANSI2KNR does not process include files, we can't + * rely on it to convert function definitions to K&R style. Hence we + * provide both styles of function header with an explicit #ifdef PROTO (ick). */ -#ifndef EOF -# define EOF (-1) -#endif /* ! EOF */ - -#ifndef NULL -# define NULL (char *)0 -#endif /* ! NULL */ +#define GVAR static /* make empty to export these variables */ /* * None of these constants are referenced in the executable portion of @@ -63,23 +69,21 @@ #define BADCH (int)'?' #define NEEDSEP (int)':' #define MAYBESEP (int)'\0' -#define ERRFD 2 #define EMSG "" #define START "-" /* * Here are all the pertinent global variables. */ -int opterr = 1; /* if true, output error message */ -int optind = 1; /* index into parent argv vector */ -int optopt; /* character checked for validity */ -int optbad = BADCH; /* character returned on error */ -int optchar = 0; /* character that begins returned option */ -int optneed = NEEDSEP; /* flag for mandatory argument */ -int optmaybe = MAYBESEP;/* flag for optional argument */ -int opterrfd = ERRFD; /* file descriptor for error text */ -char *optarg; /* argument associated with option */ -char *optstart = START; /* list of characters that start options */ +GVAR int opterr = 1; /* if true, output error message */ +GVAR int optind = 1; /* index into parent argv vector */ +GVAR int optopt; /* character checked for validity */ +GVAR int optbad = BADCH; /* character returned on error */ +GVAR int optchar = 0; /* character that begins returned option */ +GVAR int optneed = NEEDSEP; /* flag for mandatory argument */ +GVAR int optmaybe = MAYBESEP; /* flag for optional argument */ +GVAR const char *optarg; /* argument associated with option */ +GVAR const char *optstart = START; /* list of characters that start options */ /* @@ -88,17 +92,11 @@ char *optstart = START; /* list of characters that start options */ /* * Conditionally print out an error message and return (depends on the - * setting of 'opterr' and 'opterrfd'). Note that this version of - * TELL() doesn't require the existence of stdio.h. + * setting of 'opterr'). */ #define TELL(S) { \ - if (opterr && opterrfd >= 0) { \ - char option = optopt; \ - write(opterrfd, *nargv, strlen(*nargv)); \ - write(opterrfd, (S), strlen(S)); \ - write(opterrfd, &option, 1); \ - write(opterrfd, "\n", 1); \ - } \ + if (opterr) \ + fprintf(stderr, "%s%s%c\n", *nargv, (S), optopt); \ return (optbad); \ } @@ -106,10 +104,16 @@ char *optstart = START; /* list of characters that start options */ * This works similarly to index() and strchr(). I include it so that you * don't need to be concerned as to which one your system has. */ -static char * -_sindex(string, ch) -char *string; -int ch; + +#ifdef PROTO +LOCAL const char * +_sindex (const char *string, int ch) +#else +LOCAL const char * +_sindex (string, ch) + const char *string; + int ch; +#endif { if (string != NULL) { for (; *string != '\0'; ++string) { @@ -125,15 +129,21 @@ int ch; /* * Here it is: */ -int -egetopt(nargc, nargv, ostr) -int nargc; -char **nargv; -char *ostr; + +#ifdef PROTO +LOCAL int +egetopt (int nargc, char **nargv, const char *ostr) +#else +LOCAL int +egetopt (nargc, nargv, ostr) + int nargc; + char **nargv; + const char *ostr; +#endif { - static char *place = EMSG; /* option letter processing */ - register char *oli; /* option letter list index */ - register char *osi = NULL; /* option start list index */ + static const char *place = EMSG; /* option letter processing */ + register const char *oli; /* option letter list index */ + register const char *osi = NULL; /* option start list index */ if (nargv == (char **)NULL) { return (EOF); diff --git a/jbsmooth.c b/jbsmooth.c index ad2138df5..ee58c91b5 100644 --- a/jbsmooth.c +++ b/jbsmooth.c @@ -94,7 +94,7 @@ smooth_coefficients (decompress_info_ptr cinfo, #define ABS(x) ((x) < 0 ? -(x) : (x)) -#define COND_ASSIGN(_ac,_n,_z) if ((ABS(output[col][_n] - (_ac))<<1) <= Qptr[_z]) output[col][_n] = (_ac) +#define COND_ASSIGN(_ac,_n,_z) if ((ABS(output[col][_n] - (_ac))<<1) <= Qptr[_z]) output[col][_n] = (JCOEF) (_ac) COND_ASSIGN(AC01, 1, 1); COND_ASSIGN(AC02, 2, 5); diff --git a/jccolor.c b/jccolor.c index 7fd198c97..bd0e1a25f 100644 --- a/jccolor.c +++ b/jccolor.c @@ -66,11 +66,14 @@ get_rgb_ycc_rows (compress_info_ptr cinfo, * must be too; do not need an explicit range-limiting operation. */ /* Y */ - *outptr0++ = ( 306*r + 601*g + 117*b + (INT32) 512) >> 10; + *outptr0++ = (JSAMPLE) + (( 306*r + 601*g + 117*b + (INT32) 512) >> 10); /* Cb */ - *outptr1++ = ((-173)*r - 339*g + 512*b + (INT32) 512*(MAXJSAMPLE+1)) >> 10; + *outptr1++ = (JSAMPLE) + (((-173)*r - 339*g + 512*b + (INT32) 512*(MAXJSAMPLE+1)) >> 10); /* Cr */ - *outptr2++ = ( 512*r - 429*g - 83*b + (INT32) 512*(MAXJSAMPLE+1)) >> 10; + *outptr2++ = (JSAMPLE) + (( 512*r - 429*g - 83*b + (INT32) 512*(MAXJSAMPLE+1)) >> 10); } } } diff --git a/jcdeflts.c b/jcdeflts.c index cd1624ad7..150398995 100644 --- a/jcdeflts.c +++ b/jcdeflts.c @@ -6,6 +6,8 @@ * For conditions of distribution and use, see the accompanying README file. * * This file contains optional default-setting code for the JPEG compressor. + * User interfaces do not have to use this file, but those that don't use it + * must know a lot more about the innards of the JPEG code. */ #include "jinclude.h" @@ -17,11 +19,11 @@ add_huff_table (compress_info_ptr cinfo, /* Define a Huffman table */ { if (*htblptr == NULL) - *htblptr = (*cinfo->emethods->alloc_small) (SIZEOF(HUFF_TBL)); + *htblptr = (HUFF_TBL *) (*cinfo->emethods->alloc_small) (SIZEOF(HUFF_TBL)); - memcpy((void *) (*htblptr)->bits, (void *) bits, + memcpy((void *) (*htblptr)->bits, (const void *) bits, SIZEOF((*htblptr)->bits)); - memcpy((void *) (*htblptr)->huffval, (void *) val, + memcpy((void *) (*htblptr)->huffval, (const void *) val, SIZEOF((*htblptr)->huffval)); /* Initialize sent_table FALSE so table will be written to JPEG file. @@ -152,7 +154,7 @@ add_quant_table (compress_info_ptr cinfo, int which_tbl, long temp; if (*qtblptr == NULL) - *qtblptr = (*cinfo->emethods->alloc_small) (SIZEOF(QUANT_TBL)); + *qtblptr = (QUANT_TBL_PTR) (*cinfo->emethods->alloc_small) (SIZEOF(QUANT_TBL)); for (i = 0; i < DCTSIZE2; i++) { temp = ((long) basic_table[i] * scale_factor + 50L) / 100L; @@ -207,14 +209,16 @@ j_set_quality (compress_info_ptr cinfo, int quality, boolean force_baseline) * * User interfaces that don't choose to use this routine must do their * own setup of all these parameters. Alternately, you can call this - * to establish defaults and then alter parameters selectively. + * to establish defaults and then alter parameters selectively. This + * is the recommended approach since, if we add any new parameters, + * your code will still work (they'll be set to reasonable defaults). * - * See above for the meaning of the 'quality' parameter. Typically, - * the application's default quality setting will be passed to this - * routine. A later call on j_set_quality() can be used to change to - * a user-specified quality setting. + * See above for the meaning of the 'quality' and 'force_baseline' parameters. + * Typically, the application's default quality setting will be passed to this + * routine. A later call on j_set_quality() can be used to change to a + * user-specified quality setting. * - * This sets up for a color image; to output a grayscale image, + * This routine sets up for a color image; to output a grayscale image, * do this first and call j_monochrome_default() afterwards. * (The latter can be called within c_ui_method_selection, so the * choice can depend on the input file header.) @@ -223,17 +227,17 @@ j_set_quality (compress_info_ptr cinfo, int quality, boolean force_baseline) * a JFIF header (set write_JFIF_header = FALSE). * * CAUTION: if you want to compress multiple images per run, it's safest - * to call j_default_compression before *each* call to jpeg_compress (and - * j_free_defaults afterwards). If this isn't practical, you'll have to + * to call j_c_defaults before *each* call to jpeg_compress (and + * j_c_free_defaults afterwards). If this isn't practical, you'll have to * be careful to reset any individual parameters that may change during - * the compression run. The main thing you need to worry about as this - * is written is that the sent_table boolean in each Huffman table must - * be reset to FALSE before each compression; otherwise, Huffman tables - * won't get emitted for the second and subsequent images. + * the compression run. The main thing you need to worry about at present + * is that the sent_table boolean in each Huffman table must be reset to + * FALSE before each compression; otherwise, Huffman tables won't get + * emitted for the second and subsequent images. */ GLOBAL void -j_default_compression (compress_info_ptr cinfo, int quality) +j_c_defaults (compress_info_ptr cinfo, int quality, boolean force_baseline) /* NB: the external methods must already be set up. */ { short i; @@ -260,8 +264,8 @@ j_default_compression (compress_info_ptr cinfo, int quality) cinfo->write_JFIF_header = TRUE; cinfo->jpeg_color_space = CS_YCbCr; cinfo->num_components = 3; - cinfo->comp_info = (*cinfo->emethods->alloc_small) - (4 * SIZEOF(jpeg_component_info)); + cinfo->comp_info = (jpeg_component_info *) + (*cinfo->emethods->alloc_small) (4 * SIZEOF(jpeg_component_info)); /* Note: we allocate a 4-entry comp_info array so that user interface can * easily change over to CMYK color space if desired. */ @@ -294,8 +298,7 @@ j_default_compression (compress_info_ptr cinfo, int quality) compptr->ac_tbl_no = 1; /* Set up two quantization tables using the specified quality scaling */ - /* Baseline compatibility is forced (a nonissue for reasonable defaults) */ - j_set_quality(cinfo, quality, TRUE); + j_set_quality(cinfo, quality, force_baseline); /* Set up two Huffman tables in case user interface wants Huffman coding */ std_huff_tables(cinfo); @@ -327,7 +330,7 @@ j_default_compression (compress_info_ptr cinfo, int quality) GLOBAL void j_monochrome_default (compress_info_ptr cinfo) -/* Change the j_default_compression() values to emit a monochrome JPEG file. */ +/* Change the j_c_defaults() values to emit a monochrome JPEG file. */ { jpeg_component_info * compptr; @@ -341,13 +344,13 @@ j_monochrome_default (compress_info_ptr cinfo) -/* This routine releases storage allocated by j_default_compression. +/* This routine releases storage allocated by j_c_defaults. * Note that freeing the method pointer structs and the compress_info_struct * itself are the responsibility of the user interface. */ GLOBAL void -j_free_defaults (compress_info_ptr cinfo) +j_c_free_defaults (compress_info_ptr cinfo) { short i; diff --git a/jchuff.c b/jchuff.c index 531bc75ec..44a0885ab 100644 --- a/jchuff.c +++ b/jchuff.c @@ -39,8 +39,8 @@ fix_huff_tbl (HUFF_TBL * htbl) p = 0; for (l = 1; l <= 16; l++) { - for (i = 1; i <= htbl->bits[l]; i++) - huffsize[p++] = l; + for (i = 1; i <= (int) htbl->bits[l]; i++) + huffsize[p++] = (char) l; } huffsize[p] = 0; lastp = p; @@ -52,7 +52,7 @@ fix_huff_tbl (HUFF_TBL * htbl) si = huffsize[0]; p = 0; while (huffsize[p]) { - while (huffsize[p] == si) { + while (((int) huffsize[p]) == si) { huffcode[p++] = code; code++; } @@ -95,10 +95,10 @@ flush_bytes (void) } -#define emit_byte(val) ((bytes_in_buffer >= JPEG_BUF_SIZE ? \ - (flush_bytes(), 0) : 0), \ - output_buffer[bytes_in_buffer] = (val), \ - bytes_in_buffer++) +#define emit_byte(val) \ + MAKESTMT( if (bytes_in_buffer >= JPEG_BUF_SIZE) \ + flush_bytes(); \ + output_buffer[bytes_in_buffer++] = (char) (val); ) @@ -157,16 +157,22 @@ flush_bits (void) LOCAL void encode_one_block (JBLOCK block, HUFF_TBL *dctbl, HUFF_TBL *actbl) { - register INT32 temp; + register int temp, temp2; register int nbits; register int k, r, i; /* Encode the DC coefficient difference per section 7.3.5.1 */ - /* Find the number of bits needed for the magnitude of the coefficient */ - temp = block[0]; - if (temp < 0) temp = -temp; + temp = temp2 = block[0]; + + if (temp < 0) { + temp = -temp; /* temp is abs value of input */ + /* For a negative input, want temp2 = bitwise complement of abs(input) */ + /* This code assumes we are on a two's complement machine */ + temp2--; + } + /* Find the number of bits needed for the magnitude of the coefficient */ nbits = 0; while (temp) { nbits++; @@ -175,13 +181,10 @@ encode_one_block (JBLOCK block, HUFF_TBL *dctbl, HUFF_TBL *actbl) /* Emit the Huffman-coded symbol for the number of bits */ emit_bits(dctbl->ehufco[nbits], dctbl->ehufsi[nbits]); - - /* If positive, emit nbits low order bits; */ - /* if negative, emit nbits low order bits of value-1 */ - if ((temp = block[0]) < 0) - temp--; - - emit_bits((UINT16) temp, nbits); + + /* Emit that number of bits of the value, if positive, */ + /* or the complement of its magnitude, if negative. */ + emit_bits((UINT16) temp2, nbits); /* Encode the AC coefficients per section 7.3.5.2 */ @@ -196,10 +199,15 @@ encode_one_block (JBLOCK block, HUFF_TBL *dctbl, HUFF_TBL *actbl) emit_bits(actbl->ehufco[0xF0], actbl->ehufsi[0xF0]); r -= 16; } + + temp2 = temp; + if (temp < 0) { + temp = -temp; /* temp is abs value of input */ + /* This code assumes we are on a two's complement machine */ + temp2--; + } /* Find the number of bits needed for the magnitude of the coefficient */ - if (temp < 0) temp = -temp; - nbits = 1; /* there must be at least one 1 bit */ while (temp >>= 1) nbits++; @@ -208,12 +216,9 @@ encode_one_block (JBLOCK block, HUFF_TBL *dctbl, HUFF_TBL *actbl) i = (r << 4) + nbits; emit_bits(actbl->ehufco[i], actbl->ehufsi[i]); - /* If positive, emit nbits low order bits; */ - /* if negative, emit nbits low order bits of value-1 */ - if ((temp = block[k]) < 0) - temp--; - - emit_bits((UINT16) temp, nbits); + /* Emit that number of bits of the value, if positive, */ + /* or the complement of its magnitude, if negative. */ + emit_bits((UINT16) temp2, nbits); r = 0; } @@ -488,7 +493,7 @@ gen_huff_coding (compress_info_ptr cinfo, HUFF_TBL *htbl, long freq[]) for (i = 1; i <= MAX_CLEN; i++) { for (j = 0; j <= 255; j++) { if (codesize[j] == i) { - htbl->huffval[p] = j; + htbl->huffval[p] = (UINT8) j; p++; } } @@ -645,7 +650,7 @@ huff_optimize (compress_info_ptr cinfo, MCU_output_caller_ptr source_method) if (dc_count_ptrs[tbl] != NULL) { htblptr = & cinfo->dc_huff_tbl_ptrs[tbl]; if (*htblptr == NULL) - *htblptr = (*cinfo->emethods->alloc_small) (SIZEOF(HUFF_TBL)); + *htblptr = (HUFF_TBL *) (*cinfo->emethods->alloc_small) (SIZEOF(HUFF_TBL)); /* Set sent_table FALSE so updated table will be written to JPEG file. */ (*htblptr)->sent_table = FALSE; /* Compute the optimal Huffman encoding */ @@ -656,7 +661,7 @@ huff_optimize (compress_info_ptr cinfo, MCU_output_caller_ptr source_method) if (ac_count_ptrs[tbl] != NULL) { htblptr = & cinfo->ac_huff_tbl_ptrs[tbl]; if (*htblptr == NULL) - *htblptr = (*cinfo->emethods->alloc_small) (SIZEOF(HUFF_TBL)); + *htblptr = (HUFF_TBL *) (*cinfo->emethods->alloc_small) (SIZEOF(HUFF_TBL)); /* Set sent_table FALSE so updated table will be written to JPEG file. */ (*htblptr)->sent_table = FALSE; /* Compute the optimal Huffman encoding */ diff --git a/jcmain.c b/jcmain.c index f71d5adf4..3b00b409b 100644 --- a/jcmain.c +++ b/jcmain.c @@ -21,7 +21,7 @@ */ #include "jinclude.h" -#ifdef __STDC__ +#ifdef INCLUDES_ARE_ANSI #include /* to declare exit() */ #endif @@ -37,20 +37,14 @@ #define WRITE_BINARY "wb" #endif +#include "jversion.h" /* for version message */ + /* - * If your system has getopt(3), you can use your library version by - * defining HAVE_GETOPT. By default, we use the PD 'egetopt'. + * PD version of getopt(3). */ -#ifdef HAVE_GETOPT -extern int getopt PP((int argc, char **argv, char *optstring)); -extern char * optarg; -extern int optind; -#else #include "egetopt.c" -#define getopt(argc,argv,opt) egetopt(argc,argv,opt) -#endif /* @@ -58,24 +52,44 @@ extern int optind; * and selects the appropriate input-reading module. * * To determine which family of input formats the file belongs to, - * we look only at the first byte of the file, since C does not + * we may look only at the first byte of the file, since C does not * guarantee that more than one character can be pushed back with ungetc. - * This is sufficient for the currently envisioned set of input formats. - * - * If you need to look at more than one character to select an input module, - * you can either - * 1) assume you can fseek() the input file (may fail for piped input); - * 2) assume you can push back more than one character (works in + * Looking at additional bytes would require one of these approaches: + * 1) assume we can fseek() the input file (fails for piped input); + * 2) assume we can push back more than one character (works in * some C implementations, but unportable); - * or 3) don't put back the data, and modify the various input_init - * methods to assume they start reading after the start of file. + * 3) provide our own buffering as is done in djpeg (breaks input readers + * that want to use stdio directly, such as the RLE library); + * or 4) don't put back the data, and modify the input_init methods to assume + * they start reading after the start of file (also breaks RLE library). + * #1 is attractive for MS-DOS but is untenable on Unix. + * + * The most portable solution for file types that can't be identified by their + * first byte is to make the user tell us what they are. This is also the + * only approach for "raw" file types that contain only arbitrary values. + * We presently apply this method for Targa files. Most of the time Targa + * files start with 0x00, so we recognize that case. Potentially, however, + * a Targa file could start with any byte value (byte 0 is the length of the + * seldom-used ID field), so we accept a -T switch to force Targa input mode. */ +static boolean is_targa; /* records user -T switch */ + + LOCAL void select_file_type (compress_info_ptr cinfo) { int c; + if (is_targa) { +#ifdef TARGA_SUPPORTED + jselrtarga(cinfo); +#else + ERREXIT(cinfo->emethods, "Targa support was not compiled"); +#endif + return; + } + if ((c = getc(cinfo->input_file)) == EOF) ERREXIT(cinfo->emethods, "Empty input file"); @@ -89,9 +103,23 @@ select_file_type (compress_info_ptr cinfo) case 'P': jselrppm(cinfo); break; +#endif +#ifdef RLE_SUPPORTED + case 'R': + jselrrle(cinfo); + break; +#endif +#ifdef TARGA_SUPPORTED + case 0x00: + jselrtarga(cinfo); + break; #endif default: - ERREXIT(cinfo->emethods, "Unsupported input file format"); +#ifdef TARGA_SUPPORTED + ERREXIT(cinfo->emethods, "Unrecognized input file format --- did you forget -T ?"); +#else + ERREXIT(cinfo->emethods, "Unrecognized input file format"); +#endif break; } @@ -126,7 +154,7 @@ usage (char * progname) /* complain about bad command line */ { fprintf(stderr, "usage: %s ", progname); - fprintf(stderr, "[-I] [-Q quality 0..100] [-a] [-o] [-d]"); + fprintf(stderr, "[-Q quality 0..100] [-o] [-T] [-I] [-a] [-d]"); #ifdef TWO_FILE_COMMANDLINE fprintf(stderr, " inputfile outputfile\n"); #else @@ -160,18 +188,13 @@ main (int argc, char **argv) jselvirtmem(&e_methods); /* memory allocation routines */ c_methods.c_ui_method_selection = c_ui_method_selection; - /* Set up default input and output file references. */ - /* (These may be overridden below.) */ - cinfo.input_file = stdin; - cinfo.output_file = stdout; - - /* Set up default parameters. */ - e_methods.trace_level = 0; - j_default_compression(&cinfo, 75); /* default quality level */ + /* Set up default JPEG parameters. */ + j_c_defaults(&cinfo, 75, FALSE); /* default quality level = 75 */ + is_targa = FALSE; - /* Scan parameters */ + /* Scan command line options, adjust parameters */ - while ((c = getopt(argc, argv, "IQ:aod")) != EOF) + while ((c = egetopt(argc, argv, "IQ:Taod")) != EOF) switch (c) { case 'I': /* Create noninterleaved file. */ #ifdef MULTISCAN_FILES_SUPPORTED @@ -188,13 +211,16 @@ main (int argc, char **argv) usage(argv[0]); if (sscanf(optarg, "%d", &val) != 1) usage(argv[0]); - /* Note: for now, we leave force_baseline FALSE. - * In a production user interface, probably should make it TRUE - * unless overridden by a separate switch. + /* Note: for now, we make force_baseline FALSE. + * This means non-baseline JPEG files can be created with low Q values. + * To ensure only baseline files are generated, pass TRUE instead. */ j_set_quality(&cinfo, val, FALSE); } break; + case 'T': /* Input file is Targa format. */ + is_targa = TRUE; + break; case 'a': /* Use arithmetic coding. */ #ifdef ARITH_CODING_SUPPORTED cinfo.arith_code = TRUE; @@ -222,6 +248,11 @@ main (int argc, char **argv) break; } + /* If -d appeared, print version identification */ + if (e_methods.trace_level > 0) + fprintf(stderr, "Independent JPEG Group's CJPEG, version %s\n%s\n", + JVERSION, JCOPYRIGHT); + /* Select the input and output files */ #ifdef TWO_FILE_COMMANDLINE @@ -241,6 +272,9 @@ main (int argc, char **argv) #else /* not TWO_FILE_COMMANDLINE -- use Unix style */ + cinfo.input_file = stdin; /* default input file */ + cinfo.output_file = stdout; /* always the output file */ + if (optind < argc-1) { fprintf(stderr, "%s: only one input file\n", argv[0]); usage(argv[0]); @@ -261,9 +295,9 @@ main (int argc, char **argv) jpeg_compress(&cinfo); /* Release memory. */ - j_free_defaults(&cinfo); + j_c_free_defaults(&cinfo); #ifdef MEM_STATS - if (e_methods.trace_level > 0) + if (e_methods.trace_level > 0) /* Optional memory-usage statistics */ j_mem_stats(); #endif diff --git a/jconfig.h b/jconfig.h index 3b22acb73..9d6883bcc 100644 --- a/jconfig.h +++ b/jconfig.h @@ -21,7 +21,7 @@ */ /* Does your compiler support function prototypes? */ -/* (If not, you also need to use ansi2knr, see README) */ +/* (If not, you also need to use ansi2knr, see SETUP) */ #ifdef __STDC__ /* ANSI C compilers always have prototypes */ #define PROTO @@ -139,6 +139,8 @@ */ typedef int boolean; +#undef FALSE /* in case these macros already exist */ +#undef TRUE #define FALSE 0 /* values of boolean */ #define TRUE 1 @@ -171,8 +173,10 @@ typedef int boolean; #define JFIF_SUPPORTED /* JFIF or "raw JPEG" files */ #undef JTIFF_SUPPORTED /* JPEG-in-TIFF (not yet implemented) */ /* these defines indicate which image (non-JPEG) file formats are allowed */ -#define PPM_SUPPORTED /* PPM/PGM image file format */ #define GIF_SUPPORTED /* GIF image file format */ +/* #define RLE_SUPPORTED */ /* RLE image file format */ +#define PPM_SUPPORTED /* PPM/PGM image file format */ +#define TARGA_SUPPORTED /* Targa image file format */ #undef TIFF_SUPPORTED /* TIFF image file format (not yet impl.) */ /* more capability options later, no doubt */ @@ -205,6 +209,8 @@ typedef int boolean; /* First define the representation of a single pixel element value. */ #ifdef EIGHT_BIT_SAMPLES +#define BITS_IN_JSAMPLE 8 + /* JSAMPLE should be the smallest type that will hold the values 0..255. * You can use a signed char by having GETJSAMPLE mask it with 0xFF. * If you have only signed chars, and you are more worried about speed than @@ -237,6 +243,8 @@ typedef char JSAMPLE; #ifdef TWELVE_BIT_SAMPLES +#define BITS_IN_JSAMPLE 12 + /* JSAMPLE should be the smallest type that will hold the values 0..4095. */ /* On nearly all machines "short" will do nicely. */ @@ -250,6 +258,8 @@ typedef short JSAMPLE; #ifdef SIXTEEN_BIT_SAMPLES +#define BITS_IN_JSAMPLE 16 + /* JSAMPLE should be the smallest type that will hold the values 0..65535. */ #ifdef HAVE_UNSIGNED_SHORT @@ -312,9 +322,13 @@ typedef unsigned int UINT16; /* INT16 must hold at least the values -32768..32767. */ +#ifndef XMD_H /* X11/xmd.h correctly defines INT16 */ typedef short INT16; +#endif /* INT32 must hold signed 32-bit values; if your machine happens */ /* to have 64-bit longs, you might want to change this. */ +#ifndef XMD_H /* X11/xmd.h correctly defines INT32 */ typedef long INT32; +#endif diff --git a/jcpipe.c b/jcpipe.c index f58e7dbf2..a911487bb 100644 --- a/jcpipe.c +++ b/jcpipe.c @@ -248,7 +248,7 @@ subsample (compress_info_ptr cinfo, * row of whole_scan_MCUs as we can get without exceeding 64KB per row. */ -#define MAX_WHOLE_ROW_BLOCKS (65500 / SIZEOF(JBLOCK)) /* max blocks/row */ +#define MAX_WHOLE_ROW_BLOCKS ((int) (65500 / SIZEOF(JBLOCK))) /* max blocks/row */ static big_barray_ptr whole_scan_MCUs; /* Big array for saving the MCUs */ static int MCUs_in_big_row; /* # of MCUs in each row of whole_scan_MCUs */ @@ -407,7 +407,8 @@ single_ccontroller (compress_info_ptr cinfo) /* Obtain rows_this_time pixel rows and expand to rows_in_mem rows. */ /* Then we have exactly DCTSIZE row groups for subsampling. */ - rows_this_time = MIN(rows_in_mem, cinfo->image_height - cur_pixel_row); + rows_this_time = (int) MIN((long) rows_in_mem, + cinfo->image_height - cur_pixel_row); (*cinfo->methods->get_sample_rows) (cinfo, rows_this_time, fullsize_data[whichss]); @@ -569,7 +570,8 @@ single_eopt_ccontroller (compress_info_ptr cinfo) /* Obtain rows_this_time pixel rows and expand to rows_in_mem rows. */ /* Then we have exactly DCTSIZE row groups for subsampling. */ - rows_this_time = MIN(rows_in_mem, cinfo->image_height - cur_pixel_row); + rows_this_time = (int) MIN((long) rows_in_mem, + cinfo->image_height - cur_pixel_row); (*cinfo->methods->get_sample_rows) (cinfo, rows_this_time, fullsize_data[whichss]); diff --git a/jcsample.c b/jcsample.c index d23e23a09..f86034ed5 100644 --- a/jcsample.c +++ b/jcsample.c @@ -63,10 +63,10 @@ subsample (compress_info_ptr cinfo, int which_component, for (v = 0; v < v_expand; v++) { inptr = input_data[inrow+v] + (outcol*h_expand); for (h = 0; h < h_expand; h++) { - outvalue += GETJSAMPLE(*inptr++); + outvalue += (INT32) GETJSAMPLE(*inptr++); } } - *outptr++ = (outvalue + numpix2) / numpix; + *outptr++ = (JSAMPLE) ((outvalue + numpix2) / numpix); } inrow += v_expand; } diff --git a/jdcolor.c b/jdcolor.c index a39ebff2b..bc718b788 100644 --- a/jdcolor.c +++ b/jdcolor.c @@ -61,15 +61,15 @@ ycc_rgb_convert (decompress_info_ptr cinfo, int num_rows, x = y + 1436*v + 512; /* red */ if (x < 0) x = 0; if (x > ((INT32) MAXJSAMPLE*1024)) x = (INT32) MAXJSAMPLE*1024; - *outptr0++ = x >> 10; + *outptr0++ = (JSAMPLE) (x >> 10); x = y - 352*u - 731*v + 512; /* green */ if (x < 0) x = 0; if (x > ((INT32) MAXJSAMPLE*1024)) x = (INT32) MAXJSAMPLE*1024; - *outptr1++ = x >> 10; + *outptr1++ = (JSAMPLE) (x >> 10); x = y + 1815*u + 512; /* blue */ if (x < 0) x = 0; if (x > ((INT32) MAXJSAMPLE*1024)) x = (INT32) MAXJSAMPLE*1024; - *outptr2++ = x >> 10; + *outptr2++ = (JSAMPLE) (x >> 10); } } } diff --git a/jddeflts.c b/jddeflts.c new file mode 100644 index 000000000..ffcc108ea --- /dev/null +++ b/jddeflts.c @@ -0,0 +1,117 @@ +/* + * jddeflts.c + * + * Copyright (C) 1991, Thomas G. Lane. + * This file is part of the Independent JPEG Group's software. + * For conditions of distribution and use, see the accompanying README file. + * + * This file contains optional default-setting code for the JPEG decompressor. + * User interfaces do not have to use this file, but those that don't use it + * must know more about the innards of the JPEG code. + */ + +#include "jinclude.h" + + +/* + * Reload the input buffer after it's been emptied, and return the next byte. + * See the JGETC macro for calling conditions. + * + * This routine is used only if the system-dependent user interface passes + * standard_buffering = TRUE to j_d_defaults. Otherwise, the UI must supply + * a corresponding routine. Note that in any case, this routine is likely + * to be used only for JFIF or similar serial-access JPEG file formats. + * The input file control module for a random-access format such as TIFF/JPEG + * would need to override the read_jpeg_data method with its own routine. + * + * This routine would need to be replaced if reading JPEG data from something + * other than a stdio stream. + */ + +METHODDEF int +read_jpeg_data (decompress_info_ptr cinfo) +{ + cinfo->next_input_byte = cinfo->input_buffer + MIN_UNGET; + + cinfo->bytes_in_buffer = (int) FREAD(cinfo->input_file, + cinfo->next_input_byte, + JPEG_BUF_SIZE); + + if (cinfo->bytes_in_buffer <= 0) + ERREXIT(cinfo->emethods, "Unexpected EOF in JPEG file"); + + return JGETC(cinfo); +} + + + +/* Default parameter setup for decompression. + * + * User interfaces that don't choose to use this routine must do their + * own setup of all these parameters. Alternately, you can call this + * to establish defaults and then alter parameters selectively. This + * is the recommended approach since, if we add any new parameters, + * your code will still work (they'll be set to reasonable defaults). + * + * standard_buffering should be TRUE if the JPEG data is to come from + * a stdio stream and the user interface isn't interested in changing + * the normal input-buffering logic. If FALSE is passed, the user + * interface must provide its own read_jpeg_data method and must + * set up its own input buffer. (Alternately, you can pass TRUE to + * let the buffer be allocated here, then override read_jpeg_data with + * your own routine.) + */ + +GLOBAL void +j_d_defaults (decompress_info_ptr cinfo, boolean standard_buffering) +/* NB: the external methods must already be set up. */ +{ + /* Default to RGB output */ + /* UI can override by changing out_color_space */ + cinfo->out_color_space = CS_RGB; + cinfo->jpeg_color_space = CS_UNKNOWN; + /* Setting any other value in jpeg_color_space overrides heuristics in */ + /* jrdjfif.c. That might be useful when reading non-JFIF JPEG files, */ + /* but ordinarily the UI shouldn't change it. */ + + /* Default to no gamma correction of output */ + cinfo->output_gamma = 1.0; + + /* Default to no color quantization */ + cinfo->quantize_colors = FALSE; + /* but set reasonable default parameters for quantization, */ + /* so that turning on quantize_colors is sufficient to do something useful */ + cinfo->two_pass_quantize = FALSE; /* may change to TRUE later */ + cinfo->use_dithering = TRUE; + cinfo->desired_number_of_colors = 256; + + /* Default to no smoothing */ + cinfo->do_block_smoothing = FALSE; + cinfo->do_pixel_smoothing = FALSE; + + if (standard_buffering) { + /* Allocate memory for input buffer. */ + cinfo->input_buffer = (char *) (*cinfo->emethods->alloc_small) + ((size_t) (JPEG_BUF_SIZE + MIN_UNGET)); + cinfo->bytes_in_buffer = 0; /* initialize buffer to empty */ + + /* Install standard buffer-reloading method. */ + cinfo->methods->read_jpeg_data = read_jpeg_data; + } +} + + +/* This routine releases storage allocated by j_d_defaults. + * Note that freeing the method pointer structs and the decompress_info_struct + * itself are the responsibility of the user interface. + * + * standard_buffering must agree with what was passed to j_d_defaults. + */ + +GLOBAL void +j_d_free_defaults (decompress_info_ptr cinfo, boolean standard_buffering) +{ + if (standard_buffering) { + (*cinfo->emethods->free_small) ((void *) cinfo->input_buffer); + } +} diff --git a/jdhuff.c b/jdhuff.c index e9af688c5..dd429fc4d 100644 --- a/jdhuff.c +++ b/jdhuff.c @@ -17,7 +17,7 @@ static decompress_info_ptr dcinfo; -static unsigned int get_buffer; /* current bit-extraction buffer */ +static INT32 get_buffer; /* current bit-extraction buffer */ static int bits_left; /* # of unused bits in it */ @@ -35,8 +35,8 @@ fix_huff_tbl (HUFF_TBL * htbl) p = 0; for (l = 1; l <= 16; l++) { - for (i = 1; i <= htbl->bits[l]; i++) - huffsize[p++] = l; + for (i = 1; i <= (int) htbl->bits[l]; i++) + huffsize[p++] = (char) l; } huffsize[p] = 0; lastp = p; @@ -48,7 +48,7 @@ fix_huff_tbl (HUFF_TBL * htbl) si = huffsize[0]; p = 0; while (huffsize[p]) { - while (huffsize[p] == si) { + while (((int) huffsize[p]) == si) { huffcode[p++] = code; code++; } @@ -77,10 +77,11 @@ fix_huff_tbl (HUFF_TBL * htbl) htbl->maxcode[l] = -1; } } + htbl->maxcode[17] = 0xFFFFFL; /* ensures huff_DECODE terminates */ } -/* Extract the next N bits from the input stream (N <= 8) */ +/* Extract the next N bits from the input stream (N <= 15) */ LOCAL int get_bits (int nbits) @@ -90,7 +91,8 @@ get_bits (int nbits) while (nbits > bits_left) { int c = JGETC(dcinfo); - get_buffer = (get_buffer << 8) + c; + get_buffer <<= 8; + get_buffer |= c; bits_left += 8; /* If it's 0xFF, check and discard stuffed zero byte */ if (c == 0xff) { @@ -102,14 +104,14 @@ get_bits (int nbits) } bits_left -= nbits; - result = (get_buffer >> bits_left) & ((1 << nbits) - 1); + result = ((int) (get_buffer >> bits_left)) & ((1 << nbits) - 1); return result; } /* Macro to make things go at some speed! */ #define get_bit() (bits_left ? \ - ((get_buffer >> (--bits_left)) & 1) : \ + ((int) (get_buffer >> (--bits_left))) & 1 : \ get_bits(1)) @@ -127,15 +129,23 @@ huff_DECODE (HUFF_TBL * htbl) code = (code << 1) + get_bit(); l++; } + + /* With garbage input we may reach the sentinel value l = 17. */ + + if (l > 16) { + ERREXIT(dcinfo->emethods, "Corrupted data in JPEG file"); + } + + p = (int) (htbl->valptr[l] + (code - htbl->mincode[l])); - p = htbl->valptr[l] + (code - htbl->mincode[l]); - - return htbl->huffval[p]; + return (int) htbl->huffval[p]; } /* Figure 13.4.2.1.1: extend sign bit */ +/* NB: on some compilers this will only work for s > 0 */ + #define huff_EXTEND(x, s) ((x) < (1 << ((s)-1)) ? \ (x) + (-1 << (s)) + 1 : \ (x)) @@ -156,9 +166,12 @@ decode_one_block (JBLOCK block, HUFF_TBL *dctbl, HUFF_TBL *actbl) /* Section 13.4.2.1: decode the DC coefficient difference */ s = huff_DECODE(dctbl); - r = get_bits(s); - block[0] = huff_EXTEND(r, s); - + if (s) { + r = get_bits(s); + s = huff_EXTEND(r, s); + } + block[0] = s; + /* Section 13.4.2.2: decode the AC coefficients */ for (k = 1; k < DCTSIZE2; k++) { @@ -168,7 +181,7 @@ decode_one_block (JBLOCK block, HUFF_TBL *dctbl, HUFF_TBL *actbl) n = r >> 4; if (s) { - k = k + n; + k += n; r = get_bits(s); block[k] = huff_EXTEND(r, s); } else { diff --git a/jdmain.c b/jdmain.c index 109d124c9..3b60a9d8e 100644 --- a/jdmain.c +++ b/jdmain.c @@ -21,7 +21,7 @@ */ #include "jinclude.h" -#ifdef __STDC__ +#ifdef INCLUDES_ARE_ANSI #include /* to declare exit() */ #endif @@ -37,28 +37,35 @@ #define WRITE_BINARY "wb" #endif +#include "jversion.h" /* for version message */ + /* - * If your system has getopt(3), you can use your library version by - * defining HAVE_GETOPT. By default, we use the PD 'egetopt'. + * PD version of getopt(3). */ -#ifdef HAVE_GETOPT -extern int getopt PP((int argc, char **argv, char *optstring)); -extern char * optarg; -extern int optind; -#else #include "egetopt.c" -#define getopt(argc,argv,opt) egetopt(argc,argv,opt) -#endif -typedef enum { /* defines known output image formats */ - FMT_PPM, /* PPM/PGM (PBMPLUS formats) */ +/* + * This list defines the known output image formats + * (not all of which need be supported by a given version). + * You can change the default output format by defining DEFAULT_FMT; + * indeed, you had better do so if you undefine PPM_SUPPORTED. + */ + +typedef enum { FMT_GIF, /* GIF format */ + FMT_PPM, /* PPM/PGM (PBMPLUS formats) */ + FMT_RLE, /* RLE format */ + FMT_TARGA, /* Targa format */ FMT_TIFF /* TIFF format */ } IMAGE_FORMATS; +#ifndef DEFAULT_FMT /* so can override from CFLAGS in Makefile */ +#define DEFAULT_FMT FMT_PPM +#endif + static IMAGE_FORMATS requested_fmt; @@ -92,6 +99,16 @@ d_ui_method_selection (decompress_info_ptr cinfo) case FMT_PPM: jselwppm(cinfo); break; +#endif +#ifdef RLE_SUPPORTED + case FMT_RLE: + jselwrle(cinfo); + break; +#endif +#ifdef TARGA_SUPPORTED + case FMT_TARGA: + jselwtarga(cinfo); + break; #endif default: ERREXIT(cinfo->emethods, "Unsupported output file format"); @@ -100,37 +117,12 @@ d_ui_method_selection (decompress_info_ptr cinfo) } -/* - * Reload the input buffer after it's been emptied, and return the next byte. - * See the JGETC macro for calling conditions. - * - * This routine would need to be replaced if reading JPEG data from something - * other than a stdio stream. - */ - -METHODDEF int -read_jpeg_data (decompress_info_ptr cinfo) -{ - cinfo->bytes_in_buffer = fread(cinfo->input_buffer + MIN_UNGET, - 1, JPEG_BUF_SIZE, - cinfo->input_file); - - cinfo->next_input_byte = cinfo->input_buffer + MIN_UNGET; - - if (cinfo->bytes_in_buffer <= 0) - ERREXIT(cinfo->emethods, "Unexpected EOF in JPEG file"); - - return JGETC(cinfo); -} - - - LOCAL void usage (char * progname) /* complain about bad command line */ { fprintf(stderr, "usage: %s ", progname); - fprintf(stderr, "[-b] [-q colors] [-2] [-d] [-g] [-G]"); + fprintf(stderr, "[-G] [-P] [-R] [-T] [-b] [-g] [-q colors] [-2] [-D] [-d]"); #ifdef TWO_FILE_COMMANDLINE fprintf(stderr, " inputfile outputfile\n"); #else @@ -163,45 +155,36 @@ main (int argc, char **argv) jselerror(&e_methods); /* error/trace message routines */ jselvirtmem(&e_methods); /* memory allocation routines */ dc_methods.d_ui_method_selection = d_ui_method_selection; - dc_methods.read_jpeg_data = read_jpeg_data; - - /* Allocate memory for input buffer. */ - cinfo.input_buffer = (char *) (*cinfo.emethods->alloc_small) - ((size_t) (JPEG_BUF_SIZE + MIN_UNGET)); - cinfo.bytes_in_buffer = 0; /* initialize buffer to empty */ - - /* Set up default input and output file references. */ - /* (These may be overridden below.) */ - cinfo.input_file = stdin; - cinfo.output_file = stdout; - - /* Set up default parameters. */ - e_methods.trace_level = 0; - cinfo.output_gamma = 1.0; - cinfo.quantize_colors = FALSE; - cinfo.two_pass_quantize = FALSE; - cinfo.use_dithering = FALSE; - cinfo.desired_number_of_colors = 256; - cinfo.do_block_smoothing = FALSE; - cinfo.do_pixel_smoothing = FALSE; - cinfo.out_color_space = CS_RGB; - cinfo.jpeg_color_space = CS_UNKNOWN; - /* setting any other value in jpeg_color_space overrides heuristics */ - /* in jrdjfif.c ... */ - /* You may wanta change the default output format; here's the place: */ -#ifdef PPM_SUPPORTED - requested_fmt = FMT_PPM; -#else - requested_fmt = FMT_GIF; -#endif - /* Scan parameters */ + /* Set up default JPEG parameters. */ + j_d_defaults(&cinfo, TRUE); + requested_fmt = DEFAULT_FMT; /* set default output file format */ + + /* Scan command line options, adjust parameters */ - while ((c = getopt(argc, argv, "bq:2DdgG")) != EOF) + while ((c = egetopt(argc, argv, "GPRTbdgq:2D")) != EOF) switch (c) { + case 'G': /* GIF output format. */ + requested_fmt = FMT_GIF; + break; + case 'P': /* PPM output format. */ + requested_fmt = FMT_PPM; + break; + case 'R': /* RLE output format. */ + requested_fmt = FMT_RLE; + break; + case 'T': /* Targa output format. */ + requested_fmt = FMT_TARGA; + break; case 'b': /* Enable cross-block smoothing. */ cinfo.do_block_smoothing = TRUE; break; + case 'd': /* Debugging. */ + e_methods.trace_level++; + break; + case 'g': /* Force grayscale output. */ + cinfo.out_color_space = CS_GRAYSCALE; + break; case 'q': /* Do color quantization. */ { int val; if (optarg == NULL) @@ -215,17 +198,8 @@ main (int argc, char **argv) case '2': /* Use two-pass quantization. */ cinfo.two_pass_quantize = TRUE; break; - case 'D': /* Use dithering in color quantization. */ - cinfo.use_dithering = TRUE; - break; - case 'd': /* Debugging. */ - e_methods.trace_level++; - break; - case 'g': /* Force grayscale output. */ - cinfo.out_color_space = CS_GRAYSCALE; - break; - case 'G': /* GIF output format. */ - requested_fmt = FMT_GIF; + case 'D': /* Suppress dithering in color quantization. */ + cinfo.use_dithering = FALSE; break; case '?': default: @@ -233,6 +207,11 @@ main (int argc, char **argv) break; } + /* If -d appeared, print version identification */ + if (e_methods.trace_level > 0) + fprintf(stderr, "Independent JPEG Group's DJPEG, version %s\n%s\n", + JVERSION, JCOPYRIGHT); + /* Select the input and output files */ #ifdef TWO_FILE_COMMANDLINE @@ -252,6 +231,9 @@ main (int argc, char **argv) #else /* not TWO_FILE_COMMANDLINE -- use Unix style */ + cinfo.input_file = stdin; /* default input file */ + cinfo.output_file = stdout; /* always the output file */ + if (optind < argc-1) { fprintf(stderr, "%s: only one input file\n", argv[0]); usage(argv[0]); @@ -278,9 +260,9 @@ main (int argc, char **argv) jpeg_decompress(&cinfo); /* Release memory. */ - (*cinfo.emethods->free_small) ((void *) cinfo.input_buffer); + j_d_free_defaults(&cinfo, TRUE); #ifdef MEM_STATS - if (e_methods.trace_level > 0) + if (e_methods.trace_level > 0) /* Optional memory-usage statistics */ j_mem_stats(); #endif diff --git a/jdpipe.c b/jdpipe.c index 8e0887498..cdfdf2d78 100644 --- a/jdpipe.c +++ b/jdpipe.c @@ -1230,7 +1230,8 @@ multi_dcontroller (decompress_info_ptr cinfo) } emit_1pass (cinfo, - (int) MIN(rows_in_mem, cinfo->image_height-pixel_rows_output), + (int) MIN((long) rows_in_mem, + cinfo->image_height - pixel_rows_output), fullsize_ptrs, color_data); } diff --git a/jerror.c b/jerror.c index 591b1850b..f719dbcc4 100644 --- a/jerror.c +++ b/jerror.c @@ -21,7 +21,7 @@ */ #include "jinclude.h" -#ifdef __STDC__ +#ifdef INCLUDES_ARE_ANSI #include /* to declare exit() */ #endif @@ -30,7 +30,7 @@ static external_methods_ptr methods; /* saved for access to message_parm */ METHODDEF void -trace_message (char *msgtext) +trace_message (const char *msgtext) { fprintf(stderr, msgtext, methods->message_parm[0], methods->message_parm[1], @@ -42,7 +42,7 @@ trace_message (char *msgtext) METHODDEF void -error_exit (char *msgtext) +error_exit (const char *msgtext) { trace_message(msgtext); exit(1); diff --git a/jfwddct.c b/jfwddct.c index 58291ac97..21d8448b4 100644 --- a/jfwddct.c +++ b/jfwddct.c @@ -17,96 +17,145 @@ #include "jinclude.h" +/* We assume that right shift corresponds to signed division by 2 with + * rounding towards minus infinity. This is correct for typical "arithmetic + * shift" instructions that shift in copies of the sign bit. But some + * C compilers implement >> with an unsigned shift. For these machines you + * must define RIGHT_SHIFT_IS_UNSIGNED. + * RIGHT_SHIFT provides a signed right shift of an INT32 quantity. + * It is only applied with constant shift counts. + */ + +#ifdef RIGHT_SHIFT_IS_UNSIGNED +#define SHIFT_TEMPS INT32 shift_temp; +#define RIGHT_SHIFT(x,shft) \ + ((shift_temp = (x)) < 0 ? \ + (shift_temp >> (shft)) | ((~0) << (32-(shft))) : \ + (shift_temp >> (shft))) +#else +#define SHIFT_TEMPS +#define RIGHT_SHIFT(x,shft) ((x) >> (shft)) +#endif + + /* The poop on this scaling stuff is as follows: * - * Most of the numbers (after multiplication by the constants) are - * (logically) shifted left by LG2_DCT_SCALE. This is undone by UNFIXH - * before assignment to the output array. Note that we want an additional - * division by 2 on the output (required by the equations). - * - * If right shifts are unsigned, then there is a potential problem. - * However, shifting right by 16 and then assigning to a short - * (assuming short = 16 bits) will keep the sign right!! - * - * For other shifts, - * - * ((x + (1 << 30)) >> shft) - (1 << (30 - shft)) + * We have to do addition and subtraction of the integer inputs, which + * is no problem, and multiplication by fractional constants, which is + * a problem to do in integer arithmetic. We multiply all the constants + * by DCT_SCALE and convert them to integer constants (thus retaining + * LG2_DCT_SCALE bits of precision in the constants). After doing a + * multiplication we have to divide the product by DCT_SCALE, with proper + * rounding, to produce the correct output. The division can be implemented + * cheaply as a right shift of LG2_DCT_SCALE bits. The DCT equations also + * specify an additional division by 2 on the final outputs; this can be + * folded into the right-shift by shifting one more bit (see UNFIXH). * - * gives a nice right shift with sign (assuming no overflow). However, all the - * scaling is such that this isn't a problem. (Is this true?) + * If you are planning to recode this in assembler, you might want to set + * LG2_DCT_SCALE to 15. This loses a bit of precision, but then all the + * multiplications are between 16-bit quantities (given 8-bit JSAMPLEs!) + * so you could use a signed 16x16=>32 bit multiply instruction instead of + * full 32x32 multiply. Unfortunately there's no way to describe such a + * multiply portably in C, so we've gone for the extra bit of accuracy here. */ - -#define ONE 1L /* remove L if long > 32 bits */ - -#ifdef RIGHT_SHIFT_IS_UNSIGNED -#define LG2_DCT_SCALE 15 -#define RIGHT_SHIFT(_x,_shft) ((((_x) + (ONE << 30)) >> (_shft)) - (ONE << (30 - (_shft)))) -#else +#ifdef EIGHT_BIT_SAMPLES #define LG2_DCT_SCALE 16 -#define RIGHT_SHIFT(_x,_shft) ((_x) >> (_shft)) +#else +#define LG2_DCT_SCALE 15 /* lose a little precision to avoid overflow */ #endif +#define ONE ((INT32) 1) + #define DCT_SCALE (ONE << LG2_DCT_SCALE) +/* In some places we shift the inputs left by a couple more bits, */ +/* so that they can be added to fractional results without too much */ +/* loss of precision. */ #define LG2_OVERSCALE 2 -#define OVERSCALE (ONE << LG2_OVERSCALE) +#define OVERSCALE (ONE << LG2_OVERSCALE) +#define OVERSHIFT(x) ((x) <<= LG2_OVERSCALE) + +/* Scale a fractional constant by DCT_SCALE */ +#define FIX(x) ((INT32) ((x) * DCT_SCALE + 0.5)) -#define FIX(x) ((INT32) ((x) * DCT_SCALE + 0.5)) +/* Scale a fractional constant by DCT_SCALE/OVERSCALE */ +/* Such a constant can be multiplied with an overscaled input */ +/* to produce something that's scaled by DCT_SCALE */ #define FIXO(x) ((INT32) ((x) * DCT_SCALE / OVERSCALE + 0.5)) + +/* Descale and correctly round a value that's scaled by DCT_SCALE */ #define UNFIX(x) RIGHT_SHIFT((x) + (ONE << (LG2_DCT_SCALE-1)), LG2_DCT_SCALE) + +/* Same with an additional division by 2, ie, correctly rounded UNFIX(x/2) */ #define UNFIXH(x) RIGHT_SHIFT((x) + (ONE << LG2_DCT_SCALE), LG2_DCT_SCALE+1) -#define UNFIXO(x) RIGHT_SHIFT((x) + (ONE << (LG2_DCT_SCALE-1-LG2_OVERSCALE)), LG2_DCT_SCALE-LG2_OVERSCALE) -#define OVERSH(x) ((x) << LG2_OVERSCALE) -#define SIN_1_4 FIX(0.7071067811856476) +/* Take a value scaled by DCT_SCALE and round to integer scaled by OVERSCALE */ +#define UNFIXO(x) RIGHT_SHIFT((x) + (ONE << (LG2_DCT_SCALE-1-LG2_OVERSCALE)),\ + LG2_DCT_SCALE-LG2_OVERSCALE) + +/* Here are the constants we need */ +/* SIN_i_j is sine of i*pi/j, scaled by DCT_SCALE */ +/* COS_i_j is cosine of i*pi/j, scaled by DCT_SCALE */ + +#define SIN_1_4 FIX(0.707106781) #define COS_1_4 SIN_1_4 -#define SIN_1_8 FIX(0.3826834323650898) -#define COS_1_8 FIX(0.9238795325112870) +#define SIN_1_8 FIX(0.382683432) +#define COS_1_8 FIX(0.923879533) #define SIN_3_8 COS_1_8 #define COS_3_8 SIN_1_8 -#define SIN_1_16 FIX(0.1950903220161282) -#define COS_1_16 FIX(0.9807852804032300) +#define SIN_1_16 FIX(0.195090322) +#define COS_1_16 FIX(0.980785280) #define SIN_7_16 COS_1_16 #define COS_7_16 SIN_1_16 -#define SIN_3_16 FIX(0.5555702330196022) -#define COS_3_16 FIX(0.8314696123025450) +#define SIN_3_16 FIX(0.555570233) +#define COS_3_16 FIX(0.831469612) #define SIN_5_16 COS_3_16 #define COS_5_16 SIN_3_16 -#define OSIN_1_4 FIXO(0.707106781185647) +/* OSIN_i_j is sine of i*pi/j, scaled by DCT_SCALE/OVERSCALE */ +/* OCOS_i_j is cosine of i*pi/j, scaled by DCT_SCALE/OVERSCALE */ + +#define OSIN_1_4 FIXO(0.707106781) #define OCOS_1_4 OSIN_1_4 -#define OSIN_1_8 FIXO(0.3826834323650898) -#define OCOS_1_8 FIXO(0.9238795325112870) +#define OSIN_1_8 FIXO(0.382683432) +#define OCOS_1_8 FIXO(0.923879533) #define OSIN_3_8 OCOS_1_8 #define OCOS_3_8 OSIN_1_8 -#define OSIN_1_16 FIXO(0.1950903220161282) -#define OCOS_1_16 FIXO(0.9807852804032300) +#define OSIN_1_16 FIXO(0.195090322) +#define OCOS_1_16 FIXO(0.980785280) #define OSIN_7_16 OCOS_1_16 #define OCOS_7_16 OSIN_1_16 -#define OSIN_3_16 FIXO(0.5555702330196022) -#define OCOS_3_16 FIXO(0.8314696123025450) +#define OSIN_3_16 FIXO(0.555570233) +#define OCOS_3_16 FIXO(0.831469612) #define OSIN_5_16 OCOS_3_16 #define OCOS_5_16 OSIN_3_16 +/* + * Perform a 1-dimensional DCT. + * Note that this code is specialized to the case DCTSIZE = 8. + */ + INLINE LOCAL void fast_dct_8 (DCTELEM *in, int stride) { - /* tmp1x are new values of tmpx -- flashy register colourers + /* many tmps have nonoverlapping lifetime -- flashy register colourers * should be able to do this lot very well */ + INT32 in0, in1, in2, in3, in4, in5, in6, in7; INT32 tmp0, tmp1, tmp2, tmp3, tmp4, tmp5, tmp6, tmp7; - INT32 tmp10, tmp11, tmp12, tmp13, tmp14, tmp15, tmp16, tmp17; + INT32 tmp10, tmp11, tmp12, tmp13; + INT32 tmp14, tmp15, tmp16, tmp17; INT32 tmp25, tmp26; - INT32 in0, in1, in2, in3, in4, in5, in6, in7; + SHIFT_TEMPS in0 = in[ 0]; in1 = in[stride ]; @@ -126,44 +175,42 @@ fast_dct_8 (DCTELEM *in, int stride) tmp6 = in1 - in6; tmp7 = in0 - in7; - tmp10 = tmp3 + tmp0 ; - tmp11 = tmp2 + tmp1 ; - tmp12 = tmp1 - tmp2 ; - tmp13 = tmp0 - tmp3 ; - - /* Now using tmp10, tmp11, tmp12, tmp13 */ - - in[ 0] = UNFIXH((tmp10 + tmp11) * SIN_1_4); - in[stride*4] = UNFIXH((tmp10 - tmp11) * COS_1_4); + tmp10 = tmp3 + tmp0; + tmp11 = tmp2 + tmp1; + tmp12 = tmp1 - tmp2; + tmp13 = tmp0 - tmp3; - in[stride*2] = UNFIXH(tmp13*COS_1_8 + tmp12*SIN_1_8); - in[stride*6] = UNFIXH(tmp13*SIN_1_8 - tmp12*COS_1_8); + in[ 0] = (DCTELEM) UNFIXH((tmp10 + tmp11) * SIN_1_4); + in[stride*4] = (DCTELEM) UNFIXH((tmp10 - tmp11) * COS_1_4); + in[stride*2] = (DCTELEM) UNFIXH(tmp13*COS_1_8 + tmp12*SIN_1_8); + in[stride*6] = (DCTELEM) UNFIXH(tmp13*SIN_1_8 - tmp12*COS_1_8); + tmp16 = UNFIXO((tmp6 + tmp5) * SIN_1_4); tmp15 = UNFIXO((tmp6 - tmp5) * COS_1_4); + + OVERSHIFT(tmp4); + OVERSHIFT(tmp7); + + /* tmp4, tmp7, tmp15, tmp16 are overscaled by OVERSCALE */ + + tmp14 = tmp4 + tmp15; + tmp25 = tmp4 - tmp15; + tmp26 = tmp7 - tmp16; + tmp17 = tmp7 + tmp16; - /* Now using tmp10, tmp11, tmp13, tmp14, tmp15, tmp16 */ - - tmp14 = OVERSH(tmp4) + tmp15; - tmp25 = OVERSH(tmp4) - tmp15; - tmp26 = OVERSH(tmp7) - tmp16; - tmp17 = OVERSH(tmp7) + tmp16; - - /* These are now overscaled by OVERSCALE */ - - /* tmp10, tmp11, tmp12, tmp13, tmp14, tmp25, tmp26, tmp17 */ - - in[stride ] = UNFIXH(tmp17*OCOS_1_16 + tmp14*OSIN_1_16); - in[stride*7] = UNFIXH(tmp17*OCOS_7_16 - tmp14*OSIN_7_16); - in[stride*5] = UNFIXH(tmp26*OCOS_5_16 + tmp25*OSIN_5_16); - in[stride*3] = UNFIXH(tmp26*OCOS_3_16 - tmp25*OSIN_3_16); + in[stride ] = (DCTELEM) UNFIXH(tmp17*OCOS_1_16 + tmp14*OSIN_1_16); + in[stride*7] = (DCTELEM) UNFIXH(tmp17*OCOS_7_16 - tmp14*OSIN_7_16); + in[stride*5] = (DCTELEM) UNFIXH(tmp26*OCOS_5_16 + tmp25*OSIN_5_16); + in[stride*3] = (DCTELEM) UNFIXH(tmp26*OCOS_3_16 - tmp25*OSIN_3_16); } /* * Perform the forward DCT on one block of samples. * - * Note that this code is specialized to the case DCTSIZE = 8. + * A 2-D DCT can be done by 1-D DCT on each row + * followed by 1-D DCT on each column. */ GLOBAL void diff --git a/jinclude.h b/jinclude.h index ec7a7e8f8..779c61da6 100644 --- a/jinclude.h +++ b/jinclude.h @@ -13,6 +13,19 @@ */ +/* + * Normally the __STDC__ macro can be taken as indicating that the system + * include files conform to the ANSI C standard. However, if you are running + * GCC on a machine with non-ANSI system include files, that is not the case. + * In that case change the following, or add -DNONANSI_INCLUDES to your CFLAGS. + */ + +#ifdef __STDC__ +#ifndef NONANSI_INCLUDES +#define INCLUDES_ARE_ANSI /* this is what's tested before including */ +#endif +#endif + /* * is included to get the FILE typedef and NULL macro. * Note that the core portable-JPEG files do not actually do any I/O @@ -27,11 +40,15 @@ * We need the size_t typedef, which defines the parameter type of malloc(). * In an ANSI-conforming implementation this is provided by , * but on non-ANSI systems it's more likely to be in . + * On some not-quite-ANSI systems you may find it in . */ -#ifndef __STDC__ /* shouldn't need this if __STDC__ */ +#ifndef INCLUDES_ARE_ANSI /* shouldn't need this if ANSI C */ #include #endif +#ifdef __SASC /* Amiga SAS C provides it in stddef.h. */ +#include +#endif /* * In ANSI C, and indeed any rational implementation, size_t is also the @@ -41,8 +58,20 @@ * we always use this SIZEOF() macro in place of using sizeof() directly. */ +#undef SIZEOF /* in case you included X11/xmd.h */ #define SIZEOF(object) ((size_t) sizeof(object)) +/* + * fread() and fwrite() are always invoked through these macros. + * On some systems you may need to twiddle the argument casts. + * CAUTION: argument order is different from underlying functions! + */ + +#define FREAD(file,buf,sizeofbuf) \ + ((size_t) fread((void *) (buf), (size_t) 1, (size_t) (sizeofbuf), (file))) +#define FWRITE(file,buf,sizeofbuf) \ + ((size_t) fwrite((const void *) (buf), (size_t) 1, (size_t) (sizeofbuf), (file))) + /* * We need the memcpy() and strcmp() functions, plus memory zeroing. * ANSI and System V implementations declare these in . @@ -51,10 +80,10 @@ * Insert casts in these macros if not! */ -#ifdef __STDC__ +#ifdef INCLUDES_ARE_ANSI #include #define MEMZERO(voidptr,size) memset((voidptr), 0, (size)) -#else /* not STDC */ +#else /* not ANSI */ #ifdef BSD #include #define MEMZERO(voidptr,size) bzero((voidptr), (size)) @@ -63,7 +92,7 @@ #include #define MEMZERO(voidptr,size) memset((voidptr), 0, (size)) #endif /* BSD */ -#endif /* STDC */ +#endif /* ANSI */ /* Now include the portable JPEG definition files. */ diff --git a/jpegdata.h b/jpegdata.h index 3e4d75331..947a8af27 100644 --- a/jpegdata.h +++ b/jpegdata.h @@ -200,7 +200,8 @@ typedef struct { /* A Huffman coding table */ UINT16 ehufco[256]; /* code for each symbol */ char ehufsi[256]; /* length of code for each symbol */ UINT16 mincode[17]; /* smallest code of length k */ - INT32 maxcode[17]; /* largest code of length k (-1 if none) */ + INT32 maxcode[18]; /* largest code of length k (-1 if none) */ + /* maxcode[17] is a sentinel to ensure huff_DECODE terminates */ short valptr[17]; /* huffval[] index of 1st symbol of length k */ } HUFF_TBL; @@ -220,7 +221,7 @@ struct compress_info_struct { * All of these fields shall be established by the user interface before * calling jpeg_compress, or by the input_init or c_ui_method_selection * methods. - * Most parameters can be set to reasonable defaults by j_default_compression. + * Most parameters can be set to reasonable defaults by j_c_defaults. * Note that the UI must supply the storage for the main methods struct, * though it sets only a few of the methods there. */ @@ -309,8 +310,10 @@ typedef struct compress_info_struct * compress_info_ptr; struct decompress_info_struct { /* * These fields shall be established by the user interface before - * calling jpeg_decompress. Note that the UI must supply the storage for - * the main methods struct, though it sets only a few of the methods there. + * calling jpeg_decompress. + * Most parameters can be set to reasonable defaults by j_d_defaults. + * Note that the UI must supply the storage for the main methods struct, + * though it sets only a few of the methods there. */ decompress_methods_ptr methods; /* Points to list of methods to use */ @@ -425,7 +428,7 @@ typedef struct decompress_info_struct * decompress_info_ptr; #ifdef CHAR_IS_UNSIGNED #define JGETC(cinfo) ( --(cinfo)->bytes_in_buffer < 0 ? \ (*(cinfo)->methods->read_jpeg_data) (cinfo) : \ - (int) *(cinfo)->next_input_byte++ ) + (int) (*(cinfo)->next_input_byte++) ) #else #define JGETC(cinfo) ( --(cinfo)->bytes_in_buffer < 0 ? \ (*(cinfo)->methods->read_jpeg_data) (cinfo) : \ @@ -435,17 +438,34 @@ typedef struct decompress_info_struct * decompress_info_ptr; #define JUNGETC(ch,cinfo) ((cinfo)->bytes_in_buffer++, \ *(--((cinfo)->next_input_byte)) = (ch)) -#define MIN_UNGET 2 /* may always do at least 2 JUNGETCs */ +#define MIN_UNGET 4 /* may always do at least 4 JUNGETCs */ /* A virtual image has a control block whose contents are private to the * memory manager module (and may differ between managers). The rest of the - * code only refers to virtual images by these pointer types. + * code only refers to virtual images by these pointer types, and never + * dereferences the pointer. */ typedef struct big_sarray_control * big_sarray_ptr; typedef struct big_barray_control * big_barray_ptr; +/* Although a real ANSI C compiler can deal perfectly well with pointers to + * unspecified structures (see "incomplete types" in the spec), a few pre-ANSI + * and pseudo-ANSI compilers get confused. To keep one of these bozos happy, + * add -DINCOMPLETE_TYPES_BROKEN to CFLAGS in your Makefile. Then we will + * pseudo-define the structs as containing a single "dummy" field. + * The memory manager(s) #define AM_MEMORY_MANAGER before including this file, + * so that they can make their own definitions of the structs. + */ + +#ifdef INCOMPLETE_TYPES_BROKEN +#ifndef AM_MEMORY_MANAGER +struct big_sarray_control { long dummy; }; +struct big_barray_control { long dummy; }; +#endif +#endif + /* Method types that need typedefs */ @@ -490,8 +510,8 @@ struct external_methods_struct { /* by an enumerated-type code so that non-English error messages */ /* can be substituted easily. This will not be done until all the */ /* code is in place, so that we know what messages are needed. */ - METHOD(void, error_exit, (char *msgtext)); - METHOD(void, trace_message, (char *msgtext)); + METHOD(void, error_exit, (const char *msgtext)); + METHOD(void, trace_message, (const char *msgtext)); /* Working data for error/trace facility */ /* See macros below for the usage of these variables */ @@ -559,42 +579,36 @@ struct external_methods_struct { (emeth)->message_parm[3] = (p4), \ (*(emeth)->error_exit) (msg)) +#define MAKESTMT(stuff) do { stuff } while (0) + #define TRACEMS(emeth,lvl,msg) \ - ( (emeth)->trace_level >= (lvl) ? \ - ((*(emeth)->trace_message) (msg), 0) : 0) + MAKESTMT( if ((emeth)->trace_level >= (lvl)) { \ + (*(emeth)->trace_message) (msg); } ) #define TRACEMS1(emeth,lvl,msg,p1) \ - ( (emeth)->trace_level >= (lvl) ? \ - ((emeth)->message_parm[0] = (p1), \ - (*(emeth)->trace_message) (msg), 0) : 0) + MAKESTMT( if ((emeth)->trace_level >= (lvl)) { \ + (emeth)->message_parm[0] = (p1); \ + (*(emeth)->trace_message) (msg); } ) #define TRACEMS2(emeth,lvl,msg,p1,p2) \ - ( (emeth)->trace_level >= (lvl) ? \ - ((emeth)->message_parm[0] = (p1), \ - (emeth)->message_parm[1] = (p2), \ - (*(emeth)->trace_message) (msg), 0) : 0) + MAKESTMT( if ((emeth)->trace_level >= (lvl)) { \ + (emeth)->message_parm[0] = (p1); \ + (emeth)->message_parm[1] = (p2); \ + (*(emeth)->trace_message) (msg); } ) #define TRACEMS3(emeth,lvl,msg,p1,p2,p3) \ - ( (emeth)->trace_level >= (lvl) ? \ - ((emeth)->message_parm[0] = (p1), \ - (emeth)->message_parm[1] = (p2), \ - (emeth)->message_parm[2] = (p3), \ - (*(emeth)->trace_message) (msg), 0) : 0) + MAKESTMT( if ((emeth)->trace_level >= (lvl)) { \ + int * _mp = (emeth)->message_parm; \ + *_mp++ = (p1); *_mp++ = (p2); *_mp = (p3); \ + (*(emeth)->trace_message) (msg); } ) #define TRACEMS4(emeth,lvl,msg,p1,p2,p3,p4) \ - ( (emeth)->trace_level >= (lvl) ? \ - ((emeth)->message_parm[0] = (p1), \ - (emeth)->message_parm[1] = (p2), \ - (emeth)->message_parm[2] = (p3), \ - (emeth)->message_parm[3] = (p4), \ - (*(emeth)->trace_message) (msg), 0) : 0) + MAKESTMT( if ((emeth)->trace_level >= (lvl)) { \ + int * _mp = (emeth)->message_parm; \ + *_mp++ = (p1); *_mp++ = (p2); *_mp++ = (p3); *_mp = (p4); \ + (*(emeth)->trace_message) (msg); } ) #define TRACEMS8(emeth,lvl,msg,p1,p2,p3,p4,p5,p6,p7,p8) \ - ( (emeth)->trace_level >= (lvl) ? \ - ((emeth)->message_parm[0] = (p1), \ - (emeth)->message_parm[1] = (p2), \ - (emeth)->message_parm[2] = (p3), \ - (emeth)->message_parm[3] = (p4), \ - (emeth)->message_parm[4] = (p5), \ - (emeth)->message_parm[5] = (p6), \ - (emeth)->message_parm[6] = (p7), \ - (emeth)->message_parm[7] = (p8), \ - (*(emeth)->trace_message) (msg), 0) : 0) + MAKESTMT( if ((emeth)->trace_level >= (lvl)) { \ + int * _mp = (emeth)->message_parm; \ + *_mp++ = (p1); *_mp++ = (p2); *_mp++ = (p3); *_mp++ = (p4); \ + *_mp++ = (p5); *_mp++ = (p6); *_mp++ = (p7); *_mp = (p8); \ + (*(emeth)->trace_message) (msg); } ) /* Methods used during JPEG compression. */ @@ -740,16 +754,24 @@ struct decompress_methods_struct { /* main entry for compression */ EXTERN void jpeg_compress PP((compress_info_ptr cinfo)); + /* default parameter setup for compression */ -EXTERN void j_default_compression PP((compress_info_ptr cinfo, int quality)); +EXTERN void j_c_defaults PP((compress_info_ptr cinfo, int quality, + boolean force_baseline)); EXTERN void j_monochrome_default PP((compress_info_ptr cinfo)); EXTERN void j_set_quality PP((compress_info_ptr cinfo, int quality, boolean force_baseline)); -EXTERN void j_free_defaults PP((compress_info_ptr cinfo)); +EXTERN void j_c_free_defaults PP((compress_info_ptr cinfo)); /* main entry for decompression */ EXTERN void jpeg_decompress PP((decompress_info_ptr cinfo)); +/* default parameter setup for decompression */ +EXTERN void j_d_defaults PP((decompress_info_ptr cinfo, + boolean standard_buffering)); +EXTERN void j_d_free_defaults PP((decompress_info_ptr cinfo, + boolean standard_buffering)); + /* forward DCT */ EXTERN void j_fwd_dct PP((DCTBLOCK data)); /* inverse DCT */ @@ -775,6 +797,8 @@ EXTERN void jselccolor PP((compress_info_ptr cinfo)); /* jccolor.c */ /* The user interface should call one of these to select input format: */ EXTERN void jselrgif PP((compress_info_ptr cinfo)); /* jrdgif.c */ EXTERN void jselrppm PP((compress_info_ptr cinfo)); /* jrdppm.c */ +EXTERN void jselrrle PP((compress_info_ptr cinfo)); /* jrdrle.c */ +EXTERN void jselrtarga PP((compress_info_ptr cinfo)); /* jrdtarga.c */ /* and one of these to select output header format: */ EXTERN void jselwjfif PP((compress_info_ptr cinfo)); /* jwrjfif.c */ @@ -793,6 +817,8 @@ EXTERN void jselrjfif PP((decompress_info_ptr cinfo)); /* jrdjfif.c */ /* and one of these to select output image format: */ EXTERN void jselwgif PP((decompress_info_ptr cinfo)); /* jwrgif.c */ EXTERN void jselwppm PP((decompress_info_ptr cinfo)); /* jwrppm.c */ +EXTERN void jselwrle PP((decompress_info_ptr cinfo)); /* jwrrle.c */ +EXTERN void jselwtarga PP((decompress_info_ptr cinfo)); /* jwrtarga.c */ /* method selection routines for system-dependent modules */ EXTERN void jselerror PP((external_methods_ptr emethods)); /* jerror.c */ @@ -805,7 +831,9 @@ EXTERN void j_mem_stats PP((void)); /* Miscellaneous useful macros */ +#undef MAX #define MAX(a,b) ((a) > (b) ? (a) : (b)) +#undef MIN #define MIN(a,b) ((a) < (b) ? (a) : (b)) diff --git a/jquant1.c b/jquant1.c index 49dfd9717..ea01c2390 100644 --- a/jquant1.c +++ b/jquant1.c @@ -60,7 +60,7 @@ static JSAMPARRAY colorindex; /* Precomputed mapping for speed */ */ #ifdef EIGHT_BIT_SAMPLES -typedef short FSERROR; /* 16 bits should be enough */ +typedef INT16 FSERROR; /* 16 bits should be enough */ #else typedef INT32 FSERROR; /* may need more than 16 bits? */ #endif @@ -135,11 +135,13 @@ color_quant_init (decompress_info_ptr cinfo) nci = Ncolors[i]; /* # of distinct values for this color */ blksize = blkdist / nci; for (j = 0; j < nci; j++) { - val = (j * MAXJSAMPLE + (nci-1)/2) / (nci-1); /* j'th value of color */ + /* Compute j'th output value (out of nci) for component */ + val = (j * MAXJSAMPLE + (nci-1)/2) / (nci-1); + /* Fill in all colormap entries that have this value of this component */ for (ptr = j * blksize; ptr < total_colors; ptr += blkdist) { /* fill in blksize entries beginning at ptr */ for (k = 0; k < blksize; k++) - colormap[i][ptr+k] = val; + colormap[i][ptr+k] = (JSAMPLE) val; } } blkdist = blksize; /* blksize of this color is blkdist of next */ @@ -149,8 +151,7 @@ color_quant_init (decompress_info_ptr cinfo) /* compute index of color closest to pixel value j */ val = (j * (nci-1) + CENTERJSAMPLE) / MAXJSAMPLE; /* premultiply so that no multiplication needed in main processing */ - val *= blksize; - colorindex[i][j] = val; + colorindex[i][j] = (JSAMPLE) (val * blksize); } } @@ -196,7 +197,7 @@ color_quantize (decompress_info_ptr cinfo, int num_rows, pixcode += GETJSAMPLE(colorindex[ci] [GETJSAMPLE(input_data[ci][row][col])]); } - output_data[row][col] = pixcode; + output_data[row][col] = (JSAMPLE) pixcode; } } } @@ -222,7 +223,7 @@ color_quantize3 (decompress_info_ptr cinfo, int num_rows, pixcode = GETJSAMPLE(colorindex[0][GETJSAMPLE(*ptr0++)]); pixcode += GETJSAMPLE(colorindex[1][GETJSAMPLE(*ptr1++)]); pixcode += GETJSAMPLE(colorindex[2][GETJSAMPLE(*ptr2++)]); - *ptrout++ = pixcode; + *ptrout++ = (JSAMPLE) pixcode; } } } @@ -255,7 +256,7 @@ color_quantize_dither (decompress_info_ptr cinfo, int num_rows, /* compute pixel value + accumulated error */ val = (((FSERROR) GETJSAMPLE(input_data[ci][row][col])) << 4) + thisrowerr[ci]; - if (val < 0) val = 0; /* must watch for range overflow! */ + if (val <= 0) val = 0; /* must watch for range overflow! */ else { val += 8; /* divide by 16 with proper rounding */ val >>= 4; @@ -264,10 +265,10 @@ color_quantize_dither (decompress_info_ptr cinfo, int num_rows, thisrowerr[ci] = val; /* save for error propagation */ pixcode += GETJSAMPLE(colorindex[ci][val]); } - output_data[row][col] = pixcode; + output_data[row][col] = (JSAMPLE) pixcode; /* propagate error to adjacent pixels */ for (ci = 0; ci < nc; ci++) { - val = thisrowerr[ci] - GETJSAMPLE(colormap[ci][pixcode]); + val = thisrowerr[ci] - (FSERROR) GETJSAMPLE(colormap[ci][pixcode]); thisrowerr[ci-nc] += val * 7; nextrowerr[ci+nc] += val * 3; nextrowerr[ci ] += val * 5; @@ -290,7 +291,7 @@ color_quantize_dither (decompress_info_ptr cinfo, int num_rows, /* compute pixel value + accumulated error */ val = (((FSERROR) GETJSAMPLE(input_data[ci][row][col])) << 4) + thisrowerr[ci]; - if (val < 0) val = 0; /* must watch for range overflow! */ + if (val <= 0) val = 0; /* must watch for range overflow! */ else { val += 8; /* divide by 16 with proper rounding */ val >>= 4; @@ -299,10 +300,10 @@ color_quantize_dither (decompress_info_ptr cinfo, int num_rows, thisrowerr[ci] = val; /* save for error propagation */ pixcode += GETJSAMPLE(colorindex[ci][val]); } - output_data[row][col] = pixcode; + output_data[row][col] = (JSAMPLE) pixcode; /* propagate error to adjacent pixels */ for (ci = 0; ci < nc; ci++) { - val = thisrowerr[ci] - GETJSAMPLE(colormap[ci][pixcode]); + val = thisrowerr[ci] - (FSERROR) GETJSAMPLE(colormap[ci][pixcode]); thisrowerr[ci+nc] += val * 7; nextrowerr[ci-nc] += val * 3; nextrowerr[ci ] += val * 5; diff --git a/jrdgif.c b/jrdgif.c index e2d4b3381..7b6bd6d42 100644 --- a/jrdgif.c +++ b/jrdgif.c @@ -62,7 +62,7 @@ static JSAMPARRAY colormap; /* the colormap to use */ #define INTERLACE 0x40 /* mask for bit signifying interlaced image */ #define COLORMAPFLAG 0x80 /* mask for bit signifying colormap presence */ -#define ReadOK(file,buffer,len) (fread(buffer, 1, len, file) == (len)) +#define ReadOK(file,buffer,len) (FREAD(file,buffer,len) == ((size_t) (len))) /* Static vars for GetCode and LZWReadByte */ @@ -261,7 +261,7 @@ LZWReadByte (compress_info_ptr cinfo) /* If any codes are stacked from a previously read symbol, return them */ if (sp > symbol_stack) - return *(--sp); + return (int) *(--sp); code = GetCode(cinfo); @@ -286,7 +286,7 @@ LZWReadByte (compress_info_ptr cinfo) incode = code; /* save for a moment */ if (code >= max_code) { /* special case for not-yet-defined symbol */ - *sp++ = firstcode; /* it will be defined as oldcode/firstcode */ + *sp++ = (UINT8) firstcode; /* it will be defined as oldcode/firstcode */ code = oldcode; } @@ -302,7 +302,7 @@ LZWReadByte (compress_info_ptr cinfo) if ((code = max_code) < LZW_TABLE_SIZE) { /* Define a new symbol = prev sym + head of this sym's expansion */ symbol_head[code] = oldcode; - symbol_tail[code] = firstcode; + symbol_tail[code] = (UINT8) firstcode; max_code++; /* Is it time to increase code_size? */ if ((max_code >= limit_code) && (code_size < MAX_LZW_BITS)) { @@ -323,9 +323,9 @@ ReadColorMap (compress_info_ptr cinfo, int cmaplen, JSAMPARRAY cmap) int i; for (i = 0; i < cmaplen; i++) { - cmap[CM_RED][i] = ReadByte(cinfo); - cmap[CM_GREEN][i] = ReadByte(cinfo); - cmap[CM_BLUE][i] = ReadByte(cinfo); + cmap[CM_RED][i] = (JSAMPLE) ReadByte(cinfo); + cmap[CM_GREEN][i] = (JSAMPLE) ReadByte(cinfo); + cmap[CM_BLUE][i] = (JSAMPLE) ReadByte(cinfo); } } @@ -457,7 +457,7 @@ input_init (compress_info_ptr cinfo) * of get_input_row. */ interlaced_image = (*cinfo->emethods->request_big_sarray) - ((long) width, (long) height, (long) 1); + ((long) width, (long) height, 1L); cinfo->methods->get_input_row = load_interlaced_image; } @@ -519,7 +519,7 @@ load_interlaced_image (compress_info_ptr cinfo, JSAMPARRAY pixel_row) for (col = cinfo->image_width; col > 0; col--) { if ((c = LZWReadByte(cinfo)) < 0) ERREXIT(cinfo->emethods, "Premature end of GIF image"); - *sptr++ = c; + *sptr++ = (JSAMPLE) c; } } diff --git a/jrdjfif.c b/jrdjfif.c index 5ff14baa5..0729b676c 100644 --- a/jrdjfif.c +++ b/jrdjfif.c @@ -98,11 +98,11 @@ typedef enum { /* JPEG marker codes */ METHODDEF int read_jpeg_data (decompress_info_ptr cinfo) { - cinfo->bytes_in_buffer = fread(cinfo->input_buffer + MIN_UNGET, - 1, JPEG_BUF_SIZE, - cinfo->input_file); - cinfo->next_input_byte = cinfo->input_buffer + MIN_UNGET; + + cinfo->bytes_in_buffer = (int) FREAD(cinfo->input_file, + cinfo->next_input_byte, + JPEG_BUF_SIZE); if (cinfo->bytes_in_buffer <= 0) ERREXIT(cinfo->emethods, "Unexpected EOF in JPEG file"); @@ -138,7 +138,7 @@ skip_variable (decompress_info_ptr cinfo, int code) length = get_2bytes(cinfo); TRACEMS2(cinfo->emethods, 1, - "Skipping marker 0x%02x, length %d", code, length); + "Skipping marker 0x%02x, length %u", code, (int) length); for (length -= 2; length > 0; length--) (void) JGETC(cinfo); @@ -165,7 +165,7 @@ get_dht (decompress_info_ptr cinfo) bits[0] = 0; count = 0; for (i = 1; i <= 16; i++) { - bits[i] = JGETC(cinfo); + bits[i] = (UINT8) JGETC(cinfo); count += bits[i]; } @@ -180,7 +180,7 @@ get_dht (decompress_info_ptr cinfo) ERREXIT(cinfo->emethods, "Bogus DHT counts"); for (i = 0; i < count; i++) - huffval[i] = JGETC(cinfo); + huffval[i] = (UINT8) JGETC(cinfo); length -= 1 + 16 + count; @@ -195,7 +195,7 @@ get_dht (decompress_info_ptr cinfo) ERREXIT1(cinfo->emethods, "Bogus DHT index %d", index); if (*htblptr == NULL) - *htblptr = (*cinfo->emethods->alloc_small) (SIZEOF(HUFF_TBL)); + *htblptr = (HUFF_TBL *) (*cinfo->emethods->alloc_small) (SIZEOF(HUFF_TBL)); memcpy((void *) (*htblptr)->bits, (void *) bits, SIZEOF((*htblptr)->bits)); @@ -225,10 +225,10 @@ get_dac (decompress_info_ptr cinfo) ERREXIT1(cinfo->emethods, "Bogus DAC index %d", index); if (index >= NUM_ARITH_TBLS) { /* define AC table */ - cinfo->arith_ac_K[index-NUM_ARITH_TBLS] = val; + cinfo->arith_ac_K[index-NUM_ARITH_TBLS] = (UINT8) val; } else { /* define DC table */ - cinfo->arith_dc_L[index] = val & 0x0F; - cinfo->arith_dc_U[index] = val >> 4; + cinfo->arith_dc_L[index] = (UINT8) (val & 0x0F); + cinfo->arith_dc_U[index] = (UINT8) (val >> 4); if (cinfo->arith_dc_L[index] > cinfo->arith_dc_U[index]) ERREXIT1(cinfo->emethods, "Bogus DAC value 0x%x", val); } @@ -261,7 +261,8 @@ get_dqt (decompress_info_ptr cinfo) ERREXIT1(cinfo->emethods, "Bogus table number %d", n); if (cinfo->quant_tbl_ptrs[n] == NULL) - cinfo->quant_tbl_ptrs[n] = (*cinfo->emethods->alloc_small) (SIZEOF(QUANT_TBL)); + cinfo->quant_tbl_ptrs[n] = (QUANT_TBL_PTR) + (*cinfo->emethods->alloc_small) (SIZEOF(QUANT_TBL)); quant_ptr = cinfo->quant_tbl_ptrs[n]; for (i = 0; i < DCTSIZE2; i++) { @@ -290,7 +291,7 @@ get_dri (decompress_info_ptr cinfo) if (get_2bytes(cinfo) != 4) ERREXIT(cinfo->emethods, "Bogus length in DRI"); - cinfo->restart_interval = get_2bytes(cinfo); + cinfo->restart_interval = (UINT16) get_2bytes(cinfo); TRACEMS1(cinfo->emethods, 1, "Define Restart Interval %d", cinfo->restart_interval); @@ -312,7 +313,7 @@ get_app0 (decompress_info_ptr cinfo) if (length >= JFIF_LEN) { for (buffp = 0; buffp < JFIF_LEN; buffp++) - b[buffp] = JGETC(cinfo); + b[buffp] = (UINT8) JGETC(cinfo); length -= JFIF_LEN; if (b[0]=='J' && b[1]=='F' && b[2]=='I' && b[3]=='F' && b[4]==0) { @@ -364,8 +365,8 @@ get_sof (decompress_info_ptr cinfo, int code) cinfo->num_components = JGETC(cinfo); TRACEMS4(cinfo->emethods, 1, - "Start Of Frame 0x%02x: width=%d, height=%d, components=%d", - code, cinfo->image_width, cinfo->image_height, + "Start Of Frame 0x%02x: width=%u, height=%u, components=%d", + code, (int) cinfo->image_width, (int) cinfo->image_height, cinfo->num_components); /* We don't support files in which the image height is initially specified */ @@ -391,7 +392,7 @@ get_sof (decompress_info_ptr cinfo, int code) if (length != (cinfo->num_components * 3 + 8)) ERREXIT(cinfo->emethods, "Bogus SOF length"); - cinfo->comp_info = (*cinfo->emethods->alloc_small) + cinfo->comp_info = (jpeg_component_info *) (*cinfo->emethods->alloc_small) (cinfo->num_components * SIZEOF(jpeg_component_info)); for (ci = 0; ci < cinfo->num_components; ci++) { @@ -538,7 +539,7 @@ process_tables (decompress_info_ptr cinfo) case M_SOI: case M_EOI: case M_SOS: - return c; + return ((JPEG_MARKER) c); case M_DHT: get_dht(cinfo); diff --git a/jrdppm.c b/jrdppm.c index 15e2393f4..f10e0f63d 100644 --- a/jrdppm.c +++ b/jrdppm.c @@ -6,7 +6,8 @@ * For conditions of distribution and use, see the accompanying README file. * * This file contains routines to read input images in PPM format. - * The PBMPLUS library is required (well, it will be in the real version). + * The PBMPLUS library is NOT required to compile this software, + * but it is highly useful as a set of PPM image manipulation programs. * * These routines may need modification for non-Unix environments or * specialized applications. As they stand, they assume input from @@ -24,6 +25,194 @@ #ifdef PPM_SUPPORTED +static JSAMPLE * rescale; /* => maxval-remapping array, or NULL */ + + +/* Portions of this code are based on the PBMPLUS library, which is: +** +** Copyright (C) 1988 by Jef Poskanzer. +** +** Permission to use, copy, modify, and distribute this software and its +** documentation for any purpose and without fee is hereby granted, provided +** that the above copyright notice appear in all copies and that both that +** copyright notice and this permission notice appear in supporting +** documentation. This software is provided "as is" without express or +** implied warranty. +*/ + + +LOCAL int +pbm_getc (FILE * file) +/* Read next char, skipping over any comments */ +/* A comment/newline sequence is returned as a newline */ +{ + register int ch; + + ch = getc(file); + if (ch == '#') { + do { + ch = getc(file); + } while (ch != '\n' && ch != EOF); + } + return ch; +} + + +LOCAL unsigned int +read_pbm_integer (compress_info_ptr cinfo) +/* Read an unsigned decimal integer from the PPM file */ +/* Swallows one trailing character after the integer */ +/* Note that on a 16-bit-int machine, only values up to 64k can be read. */ +/* This should not be a problem in practice. */ +{ + register int ch; + register unsigned int val; + + /* Skip any leading whitespace */ + do { + ch = pbm_getc(cinfo->input_file); + if (ch == EOF) + ERREXIT(cinfo->emethods, "Premature EOF in PPM file"); + } while (ch == ' ' || ch == '\t' || ch == '\n'); + + if (ch < '0' || ch > '9') + ERREXIT(cinfo->emethods, "Bogus data in PPM file"); + + val = ch - '0'; + while ((ch = pbm_getc(cinfo->input_file)) >= '0' && ch <= '9') { + val *= 10; + val += ch - '0'; + } + return val; +} + + +/* + * Read one row of pixels. + * + * We provide several different versions depending on input file format. + * In all cases, input is scaled to the size of JSAMPLE; it's possible that + * when JSAMPLE is 12 bits, this would not really be desirable. + * + * Note that a really fast path is provided for reading raw files with + * maxval = MAXJSAMPLE, which is the normal case (at least for 8-bit JSAMPLEs). + */ + + +METHODDEF void +get_text_gray_row (compress_info_ptr cinfo, JSAMPARRAY pixel_row) +/* This version is for reading text-format PGM files with any maxval */ +{ + register JSAMPROW ptr0; + register unsigned int val; + register long col; + + ptr0 = pixel_row[0]; + for (col = cinfo->image_width; col > 0; col--) { + val = read_pbm_integer(cinfo); + if (rescale != NULL) + val = rescale[val]; + *ptr0++ = (JSAMPLE) val; + } +} + + +METHODDEF void +get_text_rgb_row (compress_info_ptr cinfo, JSAMPARRAY pixel_row) +/* This version is for reading text-format PPM files with any maxval */ +{ + register JSAMPROW ptr0, ptr1, ptr2; + register unsigned int val; + register long col; + + ptr0 = pixel_row[0]; + ptr1 = pixel_row[1]; + ptr2 = pixel_row[2]; + for (col = cinfo->image_width; col > 0; col--) { + val = read_pbm_integer(cinfo); + if (rescale != NULL) + val = rescale[val]; + *ptr0++ = (JSAMPLE) val; + val = read_pbm_integer(cinfo); + if (rescale != NULL) + val = rescale[val]; + *ptr1++ = (JSAMPLE) val; + val = read_pbm_integer(cinfo); + if (rescale != NULL) + val = rescale[val]; + *ptr2++ = (JSAMPLE) val; + } +} + + +METHODDEF void +get_scaled_gray_row (compress_info_ptr cinfo, JSAMPARRAY pixel_row) +/* This version is for reading raw-format PGM files with any maxval */ +{ + register FILE * infile = cinfo->input_file; + register JSAMPROW ptr0; + register long col; + + ptr0 = pixel_row[0]; + for (col = cinfo->image_width; col > 0; col--) { + *ptr0++ = rescale[getc(infile)]; + } +} + + +METHODDEF void +get_scaled_rgb_row (compress_info_ptr cinfo, JSAMPARRAY pixel_row) +/* This version is for reading raw-format PPM files with any maxval */ +{ + register FILE * infile = cinfo->input_file; + register JSAMPROW ptr0, ptr1, ptr2; + register long col; + + ptr0 = pixel_row[0]; + ptr1 = pixel_row[1]; + ptr2 = pixel_row[2]; + for (col = cinfo->image_width; col > 0; col--) { + *ptr0++ = rescale[getc(infile)]; + *ptr1++ = rescale[getc(infile)]; + *ptr2++ = rescale[getc(infile)]; + } +} + + +METHODDEF void +get_raw_gray_row (compress_info_ptr cinfo, JSAMPARRAY pixel_row) +/* This version is for reading raw-format PGM files with maxval = MAXJSAMPLE */ +{ + register FILE * infile = cinfo->input_file; + register JSAMPROW ptr0; + register long col; + + ptr0 = pixel_row[0]; + for (col = cinfo->image_width; col > 0; col--) { + *ptr0++ = (JSAMPLE) getc(infile); + } +} + + +METHODDEF void +get_raw_rgb_row (compress_info_ptr cinfo, JSAMPARRAY pixel_row) +/* This version is for reading raw-format PPM files with maxval = MAXJSAMPLE */ +{ + register FILE * infile = cinfo->input_file; + register JSAMPROW ptr0, ptr1, ptr2; + register long col; + + ptr0 = pixel_row[0]; + ptr1 = pixel_row[1]; + ptr2 = pixel_row[2]; + for (col = cinfo->image_width; col > 0; col--) { + *ptr0++ = (JSAMPLE) getc(infile); + *ptr1++ = (JSAMPLE) getc(infile); + *ptr2++ = (JSAMPLE) getc(infile); + } +} + + /* * Read the file header; return image size and component count. */ @@ -31,19 +220,45 @@ METHODDEF void input_init (compress_info_ptr cinfo) { - int c, w, h, prec; + int c; + unsigned int w, h, maxval; if (getc(cinfo->input_file) != 'P') ERREXIT(cinfo->emethods, "Not a PPM file"); - c = getc(cinfo->input_file); + c = getc(cinfo->input_file); /* save format discriminator for a sec */ + + w = read_pbm_integer(cinfo); /* while we fetch the header info */ + h = read_pbm_integer(cinfo); + maxval = read_pbm_integer(cinfo); + switch (c) { - case '5': /* it's a PGM file */ + case '2': /* it's a text-format PGM file */ + cinfo->methods->get_input_row = get_text_gray_row; cinfo->input_components = 1; cinfo->in_color_space = CS_GRAYSCALE; break; - case '6': /* it's a PPM file */ + case '3': /* it's a text-format PPM file */ + cinfo->methods->get_input_row = get_text_rgb_row; + cinfo->input_components = 3; + cinfo->in_color_space = CS_RGB; + break; + + case '5': /* it's a raw-format PGM file */ + if (maxval == MAXJSAMPLE) + cinfo->methods->get_input_row = get_raw_gray_row; + else + cinfo->methods->get_input_row = get_scaled_gray_row; + cinfo->input_components = 1; + cinfo->in_color_space = CS_GRAYSCALE; + break; + + case '6': /* it's a raw-format PPM file */ + if (maxval == MAXJSAMPLE) + cinfo->methods->get_input_row = get_raw_rgb_row; + else + cinfo->methods->get_input_row = get_scaled_rgb_row; cinfo->input_components = 3; cinfo->in_color_space = CS_RGB; break; @@ -53,44 +268,29 @@ input_init (compress_info_ptr cinfo) break; } - if (fscanf(cinfo->input_file, " %d %d %d", &w, &h, &prec) != 3) + if (w <= 0 || h <= 0 || maxval <= 0) /* error check */ ERREXIT(cinfo->emethods, "Not a PPM file"); - if (getc(cinfo->input_file) != '\n' || w <= 0 || h <= 0 || prec != 255) - ERREXIT(cinfo->emethods, "Not a PPM file"); + /* Compute the rescaling array if necessary */ + /* This saves per-pixel calculation */ + if (maxval == MAXJSAMPLE) + rescale = NULL; /* no rescaling required */ + else { + INT32 val, half_maxval; - cinfo->image_width = w; - cinfo->image_height = h; - cinfo->data_precision = 8; -} - - -/* - * Read one row of pixels. - */ - -METHODDEF void -get_input_row (compress_info_ptr cinfo, JSAMPARRAY pixel_row) -{ - register FILE * infile = cinfo->input_file; - register JSAMPROW ptr0, ptr1, ptr2; - register long col; - - if (cinfo->input_components == 1) { - ptr0 = pixel_row[0]; - for (col = cinfo->image_width; col > 0; col--) { - *ptr0++ = getc(infile); - } - } else { - ptr0 = pixel_row[0]; - ptr1 = pixel_row[1]; - ptr2 = pixel_row[2]; - for (col = cinfo->image_width; col > 0; col--) { - *ptr0++ = getc(infile); - *ptr1++ = getc(infile); - *ptr2++ = getc(infile); + /* On 16-bit-int machines we have to be careful of maxval = 65535 */ + rescale = (JSAMPLE *) (*cinfo->emethods->alloc_small) + ((size_t) (((long) maxval + 1L) * SIZEOF(JSAMPLE))); + half_maxval = maxval / 2; + for (val = 0; val <= (INT32) maxval; val++) { + /* The multiplication here must be done in 32 bits to avoid overflow */ + rescale[val] = (JSAMPLE) ((val * MAXJSAMPLE + half_maxval) / maxval); } } + + cinfo->image_width = w; + cinfo->image_height = h; + cinfo->data_precision = BITS_IN_JSAMPLE; } @@ -101,7 +301,8 @@ get_input_row (compress_info_ptr cinfo, JSAMPARRAY pixel_row) METHODDEF void input_term (compress_info_ptr cinfo) { - /* no work required */ + if (rescale != NULL) + (*cinfo->emethods->free_small) ((void *) rescale); } @@ -117,7 +318,7 @@ GLOBAL void jselrppm (compress_info_ptr cinfo) { cinfo->methods->input_init = input_init; - cinfo->methods->get_input_row = get_input_row; + /* cinfo->methods->get_input_row is set by input_init */ cinfo->methods->input_term = input_term; } diff --git a/jrdrle.c b/jrdrle.c new file mode 100644 index 000000000..0571d2cd8 --- /dev/null +++ b/jrdrle.c @@ -0,0 +1,372 @@ +/* + * jrdrle.c + * + * Copyright (C) 1991, Thomas G. Lane. + * This file is part of the Independent JPEG Group's software. + * For conditions of distribution and use, see the accompanying README file. + * + * This file contains routines to read input images in Utah RLE format. + * The Utah Raster Toolkit library is required (version 3.0). + * + * These routines may need modification for non-Unix environments or + * specialized applications. As they stand, they assume input from + * an ordinary stdio stream. They further assume that reading begins + * at the start of the file; input_init may need work if the + * user interface has already read some data (e.g., to determine that + * the file is indeed RLE format). + * + * These routines are invoked via the methods get_input_row + * and input_init/term. + * + * Based on code contributed by Mike Lijewski. + */ + +#include "jinclude.h" + +#ifdef RLE_SUPPORTED + +/* rle.h is provided by the Utah Raster Toolkit. */ + +#include + + +/* + * load_image assumes that JSAMPLE has the same representation as rle_pixel, + * to wit, "unsigned char". Hence we can't cope with 12- or 16-bit samples. + */ + +#ifndef EIGHT_BIT_SAMPLES + Sorry, this code only copes with 8-bit JSAMPLEs. /* deliberate syntax err */ +#endif + + +/* + * We support the following types of RLE files: + * + * GRAYSCALE - 8 bits, no colormap + * PSEUDOCOLOR - 8 bits, colormap + * TRUECOLOR - 24 bits, colormap + * DIRECTCOLOR - 24 bits, no colormap + * + * For now, we ignore any alpha channel in the image. + */ + +typedef enum { GRAYSCALE, PSEUDOCOLOR, TRUECOLOR, DIRECTCOLOR } rle_kind; + +static rle_kind visual; /* actual type of input file */ + +/* + * Since RLE stores scanlines bottom-to-top, we have to invert the image + * to conform to JPEG's top-to-bottom order. To do this, we read the + * incoming image into a virtual array on the first get_input_row call, + * then fetch the required row from the virtual array on subsequent calls. + */ + +static big_sarray_ptr image; /* single array for GRAYSCALE/PSEUDOCOLOR */ +static big_sarray_ptr red_channel; /* three arrays for TRUECOLOR/DIRECTCOLOR */ +static big_sarray_ptr green_channel; +static big_sarray_ptr blue_channel; +static long cur_row_number; /* last row# read from virtual array */ + +static rle_hdr header; /* Input file information */ +static rle_map *colormap; /* RLE colormap, if any */ + + +/* + * Read the file header; return image size and component count. + */ + +METHODDEF void +input_init (compress_info_ptr cinfo) +{ + long width, height; + + /* Use RLE library routine to get the header info */ + header.rle_file = cinfo->input_file; + switch (rle_get_setup(&header)) { + case RLE_SUCCESS: + /* A-OK */ + break; + case RLE_NOT_RLE: + ERREXIT(cinfo->emethods, "Not an RLE file"); + break; + case RLE_NO_SPACE: + ERREXIT(cinfo->emethods, "Insufficient memory for RLE header"); + break; + case RLE_EMPTY: + ERREXIT(cinfo->emethods, "Empty RLE file"); + break; + case RLE_EOF: + ERREXIT(cinfo->emethods, "Premature EOF in RLE header"); + break; + default: + ERREXIT(cinfo->emethods, "Bogus RLE error code"); + break; + } + + /* Figure out what we have, set private vars and return values accordingly */ + + width = header.xmax - header.xmin + 1; + height = header.ymax - header.ymin + 1; + header.xmin = 0; /* realign horizontally */ + header.xmax = width-1; + + cinfo->image_width = width; + cinfo->image_height = height; + cinfo->data_precision = 8; /* we can only handle 8 bit data */ + + if (header.ncolors == 1 && header.ncmap == 0) { + visual = GRAYSCALE; + TRACEMS(cinfo->emethods, 1, "Gray-scale RLE file"); + } else if (header.ncolors == 1 && header.ncmap == 3) { + visual = PSEUDOCOLOR; + colormap = header.cmap; + TRACEMS1(cinfo->emethods, 1, "Colormapped RLE file with map of length %d", + 1 << header.cmaplen); + } else if (header.ncolors == 3 && header.ncmap == 3) { + visual = TRUECOLOR; + colormap = header.cmap; + TRACEMS1(cinfo->emethods, 1, "Full-color RLE file with map of length %d", + 1 << header.cmaplen); + } else if (header.ncolors == 3 && header.ncmap == 0) { + visual = DIRECTCOLOR; + TRACEMS(cinfo->emethods, 1, "Full-color RLE file"); + } else + ERREXIT(cinfo->emethods, "Can't handle this RLE setup"); + + switch (visual) { + case GRAYSCALE: + /* request one big array to hold the grayscale image */ + image = (*cinfo->emethods->request_big_sarray) (width, height, 1L); + cinfo->in_color_space = CS_GRAYSCALE; + cinfo->input_components = 1; + break; + case PSEUDOCOLOR: + /* request one big array to hold the pseudocolor image */ + image = (*cinfo->emethods->request_big_sarray) (width, height, 1L); + cinfo->in_color_space = CS_RGB; + cinfo->input_components = 3; + break; + case TRUECOLOR: + case DIRECTCOLOR: + /* request three big arrays to hold the RGB channels */ + red_channel = (*cinfo->emethods->request_big_sarray) (width, height, 1L); + green_channel = (*cinfo->emethods->request_big_sarray) (width, height, 1L); + blue_channel = (*cinfo->emethods->request_big_sarray) (width, height, 1L); + cinfo->in_color_space = CS_RGB; + cinfo->input_components = 3; + break; + } +} + + +/* + * Read one row of pixels. + * These are called only after load_image has read the image into + * the virtual array(s). + */ + + +METHODDEF void +get_grayscale_row (compress_info_ptr cinfo, JSAMPARRAY pixel_row) +/* This is used for GRAYSCALE images */ +{ + JSAMPROW inputrows[1]; /* a pseudo JSAMPARRAY structure */ + + cur_row_number--; /* work down in array */ + + inputrows[0] = *((*cinfo->emethods->access_big_sarray) + (image, cur_row_number, FALSE)); + + jcopy_sample_rows(inputrows, 0, pixel_row, 0, 1, cinfo->image_width); +} + + +METHODDEF void +get_pseudocolor_row (compress_info_ptr cinfo, JSAMPARRAY pixel_row) +/* This is used for PSEUDOCOLOR images */ +{ + long col; + JSAMPROW image_ptr, ptr0, ptr1, ptr2; + int val; + + cur_row_number--; /* work down in array */ + + image_ptr = *((*cinfo->emethods->access_big_sarray) + (image, cur_row_number, FALSE)); + + ptr0 = pixel_row[0]; + ptr1 = pixel_row[1]; + ptr2 = pixel_row[2]; + + for (col = cinfo->image_width; col > 0; col--) { + val = GETJSAMPLE(*image_ptr++); + *ptr0++ = colormap[val ] >> 8; + *ptr1++ = colormap[val + 256] >> 8; + *ptr2++ = colormap[val + 512] >> 8; + } +} + + +METHODDEF void +get_truecolor_row (compress_info_ptr cinfo, JSAMPARRAY pixel_row) +/* This is used for TRUECOLOR images */ +/* The colormap consists of 3 independent lookup tables */ +{ + long col; + JSAMPROW red_ptr, green_ptr, blue_ptr, ptr0, ptr1, ptr2; + + cur_row_number--; /* work down in array */ + + red_ptr = *((*cinfo->emethods->access_big_sarray) + (red_channel, cur_row_number, FALSE)); + green_ptr = *((*cinfo->emethods->access_big_sarray) + (green_channel, cur_row_number, FALSE)); + blue_ptr = *((*cinfo->emethods->access_big_sarray) + (blue_channel, cur_row_number, FALSE)); + + ptr0 = pixel_row[0]; + ptr1 = pixel_row[1]; + ptr2 = pixel_row[2]; + + for (col = cinfo->image_width; col > 0; col--) { + *ptr0++ = colormap[GETJSAMPLE(*red_ptr++) ] >> 8; + *ptr1++ = colormap[GETJSAMPLE(*green_ptr++) + 256] >> 8; + *ptr2++ = colormap[GETJSAMPLE(*blue_ptr++) + 512] >> 8; + } +} + + +METHODDEF void +get_directcolor_row (compress_info_ptr cinfo, JSAMPARRAY pixel_row) +/* This is used for DIRECTCOLOR images */ +{ + JSAMPROW inputrows[3]; /* a pseudo JSAMPARRAY structure */ + + cur_row_number--; /* work down in array */ + + inputrows[0] = *((*cinfo->emethods->access_big_sarray) + (red_channel, cur_row_number, FALSE)); + inputrows[1] = *((*cinfo->emethods->access_big_sarray) + (green_channel, cur_row_number, FALSE)); + inputrows[2] = *((*cinfo->emethods->access_big_sarray) + (blue_channel, cur_row_number, FALSE)); + + jcopy_sample_rows(inputrows, 0, pixel_row, 0, 3, cinfo->image_width); +} + + +/* + * Load the color channels into separate arrays. We have to do + * this because RLE files start at the lower left while the JPEG standard + * has them starting in the upper left. This is called the first time + * we want to get a row of input. What we do is load the RLE data into + * big arrays and then call the appropriate routine to read one row from + * the big arrays. We also change cinfo->methods->get_input_row so that + * subsequent calls go straight to the row-reading routine. + */ + +METHODDEF void +load_image (compress_info_ptr cinfo, JSAMPARRAY pixel_row) +{ + long row; + rle_pixel *rle_row[3]; + + /* Read the RLE data into our virtual array(s). + * We assume here that (a) rle_pixel is represented the same as JSAMPLE, + * and (b) we are not on a machine where FAR pointers differ from regular. + */ + RLE_CLR_BIT(header, RLE_ALPHA); /* don't read the alpha channel */ + + switch (visual) { + case GRAYSCALE: + case PSEUDOCOLOR: + for (row = 0; row < cinfo->image_height; row++) { + /* + * Read a row of the image directly into our big array. + * Too bad this doesn't seem to return any indication of errors :-(. + */ + rle_row[0] = (rle_pixel *) *((*cinfo->emethods->access_big_sarray) + (image, row, TRUE)); + rle_getrow(&header, rle_row); + } + break; + case TRUECOLOR: + case DIRECTCOLOR: + for (row = 0; row < cinfo->image_height; row++) { + /* + * Read a row of the image directly into our big arrays. + * Too bad this doesn't seem to return any indication of errors :-(. + */ + rle_row[0] = (rle_pixel *) *((*cinfo->emethods->access_big_sarray) + (red_channel, row, TRUE)); + rle_row[1] = (rle_pixel *) *((*cinfo->emethods->access_big_sarray) + (green_channel, row, TRUE)); + rle_row[2] = (rle_pixel *) *((*cinfo->emethods->access_big_sarray) + (blue_channel, row, TRUE)); + rle_getrow(&header, rle_row); + } + break; + } + + /* Set up to call proper row-extraction routine in future */ + switch (visual) { + case GRAYSCALE: + cinfo->methods->get_input_row = get_grayscale_row; + break; + case PSEUDOCOLOR: + cinfo->methods->get_input_row = get_pseudocolor_row; + break; + case TRUECOLOR: + cinfo->methods->get_input_row = get_truecolor_row; + break; + case DIRECTCOLOR: + cinfo->methods->get_input_row = get_directcolor_row; + break; + } + + /* And fetch the topmost (bottommost) row */ + cur_row_number = cinfo->image_height; + (*cinfo->methods->get_input_row) (cinfo, pixel_row); +} + + +/* + * Finish up at the end of the file. + */ + +METHODDEF void +input_term (compress_info_ptr cinfo) +{ + switch (visual) { + case GRAYSCALE: + case PSEUDOCOLOR: + (*cinfo->emethods->free_big_sarray) (image); + break; + case TRUECOLOR: + case DIRECTCOLOR: + (*cinfo->emethods->free_big_sarray) (red_channel); + (*cinfo->emethods->free_big_sarray) (green_channel); + (*cinfo->emethods->free_big_sarray) (blue_channel); + break; + } +} + + +/* + * The method selection routine for RLE format input. + * Note that this must be called by the user interface before calling + * jpeg_compress. If multiple input formats are supported, the + * user interface is responsible for discovering the file format and + * calling the appropriate method selection routine. + */ + +GLOBAL void +jselrrle (compress_info_ptr cinfo) +{ + cinfo->methods->input_init = input_init; + cinfo->methods->get_input_row = load_image; /* until first call */ + cinfo->methods->input_term = input_term; +} + +#endif /* RLE_SUPPORTED */ diff --git a/jrdtarga.c b/jrdtarga.c new file mode 100644 index 000000000..b8aa03320 --- /dev/null +++ b/jrdtarga.c @@ -0,0 +1,483 @@ +/* + * jrdtarga.c + * + * Copyright (C) 1991, Thomas G. Lane. + * This file is part of the Independent JPEG Group's software. + * For conditions of distribution and use, see the accompanying README file. + * + * This file contains routines to read input images in Targa format. + * + * These routines may need modification for non-Unix environments or + * specialized applications. As they stand, they assume input from + * an ordinary stdio stream. They further assume that reading begins + * at the start of the file; input_init may need work if the + * user interface has already read some data (e.g., to determine that + * the file is indeed Targa format). + * + * These routines are invoked via the methods get_input_row + * and input_init/term. + * + * Based on code contributed by Lee Daniel Crocker. + */ + +#include "jinclude.h" + +#ifdef TARGA_SUPPORTED + + +/* Macros to deal with unsigned chars as efficiently as compiler allows */ + +#ifdef HAVE_UNSIGNED_CHAR +typedef unsigned char U_CHAR; +#define UCH(x) ((int) (x)) +#else /* !HAVE_UNSIGNED_CHAR */ +#ifdef CHAR_IS_UNSIGNED +typedef char U_CHAR; +#define UCH(x) ((int) (x)) +#else +typedef char U_CHAR; +#define UCH(x) ((int) (x) & 0xFF) +#endif +#endif /* HAVE_UNSIGNED_CHAR */ + + +#define ReadOK(file,buffer,len) (FREAD(file,buffer,len) == ((size_t) (len))) + + +static JSAMPARRAY colormap; /* Targa colormap (converted to my format) */ + +static big_sarray_ptr whole_image; /* Needed if funny input row order */ +static long current_row; /* Current logical row number to read */ + +/* Pointer to routine to extract next Targa pixel from input file */ +static void (*read_pixel) PP((compress_info_ptr cinfo)); + +/* Result of read_pixel is delivered here: */ +static U_CHAR tga_pixel[4]; + +static int pixel_size; /* Bytes per Targa pixel (1 to 4) */ + +/* State info for reading RLE-coded pixels; both counts must be init to 0 */ +static int block_count; /* # of pixels remaining in RLE block */ +static int dup_pixel_count; /* # of times to duplicate previous pixel */ + +/* This saves the correct pixel-row-expansion method for preload_image */ +static void (*get_pixel_row) PP((compress_info_ptr cinfo, + JSAMPARRAY pixel_row)); + + +/* For expanding 5-bit pixel values to 8-bit with best rounding */ + +static const UINT8 c5to8bits[32] = { + 0, 8, 16, 24, 32, 41, 49, 57, + 65, 74, 82, 90, 98, 106, 115, 123, + 131, 139, 148, 156, 164, 172, 180, 189, + 197, 205, 213, 222, 230, 238, 246, 255 +}; + + + +LOCAL int +read_byte (compress_info_ptr cinfo) +/* Read next byte from Targa file */ +{ + register FILE *infile = cinfo->input_file; + register int c; + + if ((c = getc(infile)) == EOF) + ERREXIT(cinfo->emethods, "Premature EOF in Targa file"); + return c; +} + + +LOCAL void +read_colormap (compress_info_ptr cinfo, int cmaplen, int mapentrysize) +/* Read the colormap from a Targa file */ +{ + int i; + + /* Presently only handles 24-bit BGR format */ + if (mapentrysize != 24) + ERREXIT(cinfo->emethods, "Unsupported Targa colormap format"); + + for (i = 0; i < cmaplen; i++) { + colormap[2][i] = (JSAMPLE) read_byte(cinfo); + colormap[1][i] = (JSAMPLE) read_byte(cinfo); + colormap[0][i] = (JSAMPLE) read_byte(cinfo); + } +} + + +/* + * read_pixel methods: get a single pixel from Targa file into tga_pixel[] + */ + +LOCAL void +read_non_rle_pixel (compress_info_ptr cinfo) +/* Read one Targa pixel from the input file; no RLE expansion */ +{ + register FILE * infile = cinfo->input_file; + register int i; + + for (i = 0; i < pixel_size; i++) { + tga_pixel[i] = (U_CHAR) getc(infile); + } +} + + +LOCAL void +read_rle_pixel (compress_info_ptr cinfo) +/* Read one Targa pixel from the input file, expanding RLE data as needed */ +{ + register FILE * infile = cinfo->input_file; + register int i; + + /* Duplicate previously read pixel? */ + if (dup_pixel_count > 0) { + dup_pixel_count--; + return; + } + + /* Time to read RLE block header? */ + if (--block_count < 0) { /* decrement pixels remaining in block */ + i = read_byte(cinfo); + if (i & 0x80) { /* Start of duplicate-pixel block? */ + dup_pixel_count = i & 0x7F; /* number of duplications after this one */ + block_count = 0; /* then read new block header */ + } else { + block_count = i & 0x7F; /* number of pixels after this one */ + } + } + + /* Read next pixel */ + for (i = 0; i < pixel_size; i++) { + tga_pixel[i] = (U_CHAR) getc(infile); + } +} + + +/* + * Read one row of pixels. + * + * We provide several different versions depending on input file format. + */ + + +METHODDEF void +get_8bit_gray_row (compress_info_ptr cinfo, JSAMPARRAY pixel_row) +/* This version is for reading 8-bit grayscale pixels */ +{ + register JSAMPROW ptr0; + register long col; + + ptr0 = pixel_row[0]; + for (col = cinfo->image_width; col > 0; col--) { + (*read_pixel) (cinfo); /* Load next pixel into tga_pixel */ + *ptr0++ = (JSAMPLE) UCH(tga_pixel[0]); + } +} + +METHODDEF void +get_8bit_row (compress_info_ptr cinfo, JSAMPARRAY pixel_row) +/* This version is for reading 8-bit colormap indexes */ +{ + register int t; + register JSAMPROW ptr0, ptr1, ptr2; + register long col; + + ptr0 = pixel_row[0]; + ptr1 = pixel_row[1]; + ptr2 = pixel_row[2]; + for (col = cinfo->image_width; col > 0; col--) { + (*read_pixel) (cinfo); /* Load next pixel into tga_pixel */ + t = UCH(tga_pixel[0]); + *ptr0++ = colormap[0][t]; + *ptr1++ = colormap[1][t]; + *ptr2++ = colormap[2][t]; + } +} + +METHODDEF void +get_16bit_row (compress_info_ptr cinfo, JSAMPARRAY pixel_row) +/* This version is for reading 16-bit pixels */ +{ + register int t; + register JSAMPROW ptr0, ptr1, ptr2; + register long col; + + ptr0 = pixel_row[0]; + ptr1 = pixel_row[1]; + ptr2 = pixel_row[2]; + for (col = cinfo->image_width; col > 0; col--) { + (*read_pixel) (cinfo); /* Load next pixel into tga_pixel */ + t = UCH(tga_pixel[0]); + t += UCH(tga_pixel[1]) << 8; + /* We expand 5 bit data to 8 bit sample width. + * The format of the 16-bit (LSB first) input word is + * xRRRRRGGGGGBBBBB + */ + *ptr2++ = (JSAMPLE) c5to8bits[t & 0x1F]; + t >>= 5; + *ptr1++ = (JSAMPLE) c5to8bits[t & 0x1F]; + t >>= 5; + *ptr0++ = (JSAMPLE) c5to8bits[t & 0x1F]; + } +} + +METHODDEF void +get_24bit_row (compress_info_ptr cinfo, JSAMPARRAY pixel_row) +/* This version is for reading 24-bit pixels */ +{ + register JSAMPROW ptr0, ptr1, ptr2; + register long col; + + ptr0 = pixel_row[0]; + ptr1 = pixel_row[1]; + ptr2 = pixel_row[2]; + for (col = cinfo->image_width; col > 0; col--) { + (*read_pixel) (cinfo); /* Load next pixel into tga_pixel */ + *ptr0++ = (JSAMPLE) UCH(tga_pixel[2]); /* convert BGR to RGB order */ + *ptr1++ = (JSAMPLE) UCH(tga_pixel[1]); + *ptr2++ = (JSAMPLE) UCH(tga_pixel[0]); + } +} + +METHODDEF void +get_32bit_row (compress_info_ptr cinfo, JSAMPARRAY pixel_row) +/* This version is for reading 32-bit pixels */ +/* Attribute bits are ignored for now */ +{ + register JSAMPROW ptr0, ptr1, ptr2; + register long col; + +/* NOTE: there seems to be considerable confusion over whether the order + * of the bytes in a 32-bit Targa file is A,B,G,R or B,G,R,A. + * On Lee Crocker's authority, we think the attribute byte comes first. + * Make ATTR_BYTE_FIRST be 0 if you have files in which it comes last. + */ +#ifndef ATTR_BYTE_FIRST /* so you can say -DATTR_BYTE_FIRST=0 in Makefile */ +#define ATTR_BYTE_FIRST 1 /* must be 0 or 1 */ +#endif + + ptr0 = pixel_row[0]; + ptr1 = pixel_row[1]; + ptr2 = pixel_row[2]; + for (col = cinfo->image_width; col > 0; col--) { + (*read_pixel) (cinfo); /* Load next pixel into tga_pixel */ + /* convert ABGR (or BGRA) to RGB order */ + *ptr0++ = (JSAMPLE) UCH(tga_pixel[2+ATTR_BYTE_FIRST]); + *ptr1++ = (JSAMPLE) UCH(tga_pixel[1+ATTR_BYTE_FIRST]); + *ptr2++ = (JSAMPLE) UCH(tga_pixel[0+ATTR_BYTE_FIRST]); + } +} + + +/* + * This method is for re-reading the input data in standard top-down + * row order. The entire image has already been read into whole_image + * with proper conversion of pixel format, but it's in a funny row order. + */ + +METHODDEF void +get_memory_row (compress_info_ptr cinfo, JSAMPARRAY pixel_row) +{ + JSAMPARRAY image_ptr; + long source_row; + + /* Compute row of source that maps to current_row of normal order */ + /* For now, assume image is bottom-up and not interlaced. */ + /* NEEDS WORK to support interlaced images! */ + source_row = cinfo->image_height - current_row - 1; + + /* Fetch that row from virtual array */ + image_ptr = (*cinfo->emethods->access_big_sarray) + (whole_image, source_row * cinfo->input_components, FALSE); + + jcopy_sample_rows(image_ptr, 0, pixel_row, 0, + cinfo->input_components, cinfo->image_width); + + current_row++; +} + + +/* + * This method loads the image into whole_image during the first call on + * get_input_row. The get_input_row pointer is then adjusted to call + * get_memory_row on subsequent calls. + */ + +METHODDEF void +preload_image (compress_info_ptr cinfo, JSAMPARRAY pixel_row) +{ + JSAMPARRAY image_ptr; + long row; + + /* Read the data into a virtual array in input-file row order */ + for (row = 0; row < cinfo->image_height; row++) { + image_ptr = (*cinfo->emethods->access_big_sarray) + (whole_image, row * cinfo->input_components, TRUE); + (*get_pixel_row) (cinfo, image_ptr); + } + /* Set up to read from the virtual array in unscrambled order */ + cinfo->methods->get_input_row = get_memory_row; + current_row = 0; + /* And read the first row */ + get_memory_row(cinfo, pixel_row); +} + + +/* + * Read the file header; return image size and component count. + */ + +METHODDEF void +input_init (compress_info_ptr cinfo) +{ + U_CHAR targaheader[18]; + int idlen, cmaptype, subtype, flags, interlace_type, components; + UINT16 width, height, maplen; + boolean is_bottom_up; + +#define GET_2B(offset) ((unsigned int) UCH(targaheader[offset]) + \ + (((unsigned int) UCH(targaheader[offset+1])) << 8)) + + if (! ReadOK(cinfo->input_file, targaheader, 18)) + ERREXIT(cinfo->emethods, "Unexpected end of file"); + + /* Pretend "15-bit" pixels are 16-bit --- we ignore attribute bit anyway */ + if (targaheader[16] == 15) + targaheader[16] = 16; + + idlen = UCH(targaheader[0]); + cmaptype = UCH(targaheader[1]); + subtype = UCH(targaheader[2]); + maplen = GET_2B(5); + width = GET_2B(12); + height = GET_2B(14); + pixel_size = UCH(targaheader[16]) >> 3; + flags = UCH(targaheader[17]); /* Image Descriptor byte */ + + is_bottom_up = ((flags & 0x20) == 0); /* bit 5 set => top-down */ + interlace_type = flags >> 6; /* bits 6/7 are interlace code */ + + if (cmaptype > 1 || /* cmaptype must be 0 or 1 */ + pixel_size < 1 || pixel_size > 4 || + (UCH(targaheader[16]) & 7) != 0 || /* bits/pixel must be multiple of 8 */ + interlace_type != 0) /* currently don't allow interlaced image */ + ERREXIT(cinfo->emethods, "Invalid or unsupported Targa file"); + + if (subtype > 8) { + /* It's an RLE-coded file */ + read_pixel = read_rle_pixel; + block_count = dup_pixel_count = 0; + subtype -= 8; + } else { + /* Non-RLE file */ + read_pixel = read_non_rle_pixel; + } + + /* Now should have subtype 1, 2, or 3 */ + components = 3; /* until proven different */ + cinfo->in_color_space = CS_RGB; + + switch (subtype) { + case 1: /* colormapped image */ + if (pixel_size == 1 && cmaptype == 1) + get_pixel_row = get_8bit_row; + else + ERREXIT(cinfo->emethods, "Invalid or unsupported Targa file"); + break; + case 2: /* RGB image */ + switch (pixel_size) { + case 2: + get_pixel_row = get_16bit_row; + break; + case 3: + get_pixel_row = get_24bit_row; + break; + case 4: + get_pixel_row = get_32bit_row; + break; + default: + ERREXIT(cinfo->emethods, "Invalid or unsupported Targa file"); + break; + } + break; + case 3: /* Grayscale image */ + components = 1; + cinfo->in_color_space = CS_GRAYSCALE; + if (pixel_size == 1) + get_pixel_row = get_8bit_gray_row; + else + ERREXIT(cinfo->emethods, "Invalid or unsupported Targa file"); + break; + default: + ERREXIT(cinfo->emethods, "Invalid or unsupported Targa file"); + break; + } + + if (is_bottom_up) { + whole_image = (*cinfo->emethods->request_big_sarray) + ((long) width, (long) height * components, + (long) components); + cinfo->methods->get_input_row = preload_image; + } else { + whole_image = NULL; + cinfo->methods->get_input_row = get_pixel_row; + } + + while (idlen--) /* Throw away ID field */ + (void) read_byte(cinfo); + + if (maplen > 0) { + if (maplen > 256 || GET_2B(3) != 0) + ERREXIT(cinfo->emethods, "Colormap too large"); + /* Allocate space to store the colormap */ + colormap = (*cinfo->emethods->alloc_small_sarray) + ((long) maplen, 3L); + /* and read it from the file */ + read_colormap(cinfo, (int) maplen, UCH(targaheader[7])); + } else { + if (cmaptype) /* but you promised a cmap! */ + ERREXIT(cinfo->emethods, "Invalid or unsupported Targa file"); + colormap = NULL; + } + + cinfo->input_components = components; + cinfo->image_width = width; + cinfo->image_height = height; + cinfo->data_precision = 8; /* always, even if 12-bit JSAMPLEs */ +} + + +/* + * Finish up at the end of the file. + */ + +METHODDEF void +input_term (compress_info_ptr cinfo) +{ + if (whole_image != NULL) + (*cinfo->emethods->free_big_sarray) (whole_image); + if (colormap != NULL) + (*cinfo->emethods->free_small_sarray) (colormap, 3L); +} + + +/* + * The method selection routine for Targa format input. + * Note that this must be called by the user interface before calling + * jpeg_compress. If multiple input formats are supported, the + * user interface is responsible for discovering the file format and + * calling the appropriate method selection routine. + */ + +GLOBAL void +jselrtarga (compress_info_ptr cinfo) +{ + cinfo->methods->input_init = input_init; + /* cinfo->methods->get_input_row is set by input_init */ + cinfo->methods->input_term = input_term; +} + +#endif /* TARGA_SUPPORTED */ diff --git a/jrevdct.c b/jrevdct.c index bafbf5516..7c6d83ff5 100644 --- a/jrevdct.c +++ b/jrevdct.c @@ -16,99 +16,147 @@ #include "jinclude.h" +/* We assume that right shift corresponds to signed division by 2 with + * rounding towards minus infinity. This is correct for typical "arithmetic + * shift" instructions that shift in copies of the sign bit. But some + * C compilers implement >> with an unsigned shift. For these machines you + * must define RIGHT_SHIFT_IS_UNSIGNED. + * RIGHT_SHIFT provides a signed right shift of an INT32 quantity. + * It is only applied with constant shift counts. + */ + +#ifdef RIGHT_SHIFT_IS_UNSIGNED +#define SHIFT_TEMPS INT32 shift_temp; +#define RIGHT_SHIFT(x,shft) \ + ((shift_temp = (x)) < 0 ? \ + (shift_temp >> (shft)) | ((~0) << (32-(shft))) : \ + (shift_temp >> (shft))) +#else +#define SHIFT_TEMPS +#define RIGHT_SHIFT(x,shft) ((x) >> (shft)) +#endif + + /* The poop on this scaling stuff is as follows: * - * Most of the numbers (after multiplication by the constants) are - * (logically) shifted left by LG2_DCT_SCALE. This is undone by UNFIXH - * before assignment to the output array. Note that we want an additional - * division by 2 on the output (required by the equations). - * - * If right shifts are unsigned, then there is a potential problem. - * However, shifting right by 16 and then assigning to a short - * (assuming short = 16 bits) will keep the sign right!! - * - * For other shifts, + * We have to do addition and subtraction of the integer inputs, which + * is no problem, and multiplication by fractional constants, which is + * a problem to do in integer arithmetic. We multiply all the constants + * by DCT_SCALE and convert them to integer constants (thus retaining + * LG2_DCT_SCALE bits of precision in the constants). After doing a + * multiplication we have to divide the product by DCT_SCALE, with proper + * rounding, to produce the correct output. The division can be implemented + * cheaply as a right shift of LG2_DCT_SCALE bits. The DCT equations also + * specify an additional division by 2 on the final outputs; this can be + * folded into the right-shift by shifting one more bit (see UNFIXH). * - * ((x + (1 << 30)) >> shft) - (1 << (30 - shft)) - * - * gives a nice right shift with sign (assuming no overflow). However, all the - * scaling is such that this isn't a problem. (Is this true?) + * If you are planning to recode this in assembler, you might want to set + * LG2_DCT_SCALE to 15. This loses a bit of precision, but then all the + * multiplications are between 16-bit quantities (given 8-bit JSAMPLEs!) + * so you could use a signed 16x16=>32 bit multiply instruction instead of + * full 32x32 multiply. Unfortunately there's no way to describe such a + * multiply portably in C, so we've gone for the extra bit of accuracy here. */ - -#define ONE 1L /* remove L if long > 32 bits */ - -#ifdef RIGHT_SHIFT_IS_UNSIGNED -#define LG2_DCT_SCALE 15 -#define RIGHT_SHIFT(_x,_shft) ((((_x) + (ONE << 30)) >> (_shft)) - (ONE << (30 - (_shft)))) -#else +#ifdef EIGHT_BIT_SAMPLES #define LG2_DCT_SCALE 16 -#define RIGHT_SHIFT(_x,_shft) ((_x) >> (_shft)) +#else +#define LG2_DCT_SCALE 15 /* lose a little precision to avoid overflow */ #endif +#define ONE ((INT32) 1) + #define DCT_SCALE (ONE << LG2_DCT_SCALE) +/* In some places we shift the inputs left by a couple more bits, */ +/* so that they can be added to fractional results without too much */ +/* loss of precision. */ #define LG2_OVERSCALE 2 -#define OVERSCALE (ONE << LG2_OVERSCALE) +#define OVERSCALE (ONE << LG2_OVERSCALE) +#define OVERSHIFT(x) ((x) <<= LG2_OVERSCALE) + +/* Scale a fractional constant by DCT_SCALE */ +#define FIX(x) ((INT32) ((x) * DCT_SCALE + 0.5)) -#define FIX(x) ((INT32) ((x) * DCT_SCALE + 0.5)) +/* Scale a fractional constant by DCT_SCALE/OVERSCALE */ +/* Such a constant can be multiplied with an overscaled input */ +/* to produce something that's scaled by DCT_SCALE */ #define FIXO(x) ((INT32) ((x) * DCT_SCALE / OVERSCALE + 0.5)) + +/* Descale and correctly round a value that's scaled by DCT_SCALE */ #define UNFIX(x) RIGHT_SHIFT((x) + (ONE << (LG2_DCT_SCALE-1)), LG2_DCT_SCALE) + +/* Same with an additional division by 2, ie, correctly rounded UNFIX(x/2) */ #define UNFIXH(x) RIGHT_SHIFT((x) + (ONE << LG2_DCT_SCALE), LG2_DCT_SCALE+1) -#define UNFIXO(x) RIGHT_SHIFT((x) + (ONE << (LG2_DCT_SCALE-1-LG2_OVERSCALE)), LG2_DCT_SCALE-LG2_OVERSCALE) -#define OVERSH(x) ((x) << LG2_OVERSCALE) -#define SIN_1_4 FIX(0.7071067811856476) +/* Take a value scaled by DCT_SCALE and round to integer scaled by OVERSCALE */ +#define UNFIXO(x) RIGHT_SHIFT((x) + (ONE << (LG2_DCT_SCALE-1-LG2_OVERSCALE)),\ + LG2_DCT_SCALE-LG2_OVERSCALE) + +/* Here are the constants we need */ +/* SIN_i_j is sine of i*pi/j, scaled by DCT_SCALE */ +/* COS_i_j is cosine of i*pi/j, scaled by DCT_SCALE */ + +#define SIN_1_4 FIX(0.707106781) #define COS_1_4 SIN_1_4 -#define SIN_1_8 FIX(0.3826834323650898) -#define COS_1_8 FIX(0.9238795325112870) +#define SIN_1_8 FIX(0.382683432) +#define COS_1_8 FIX(0.923879533) #define SIN_3_8 COS_1_8 #define COS_3_8 SIN_1_8 -#define SIN_1_16 FIX(0.1950903220161282) -#define COS_1_16 FIX(0.9807852804032300) +#define SIN_1_16 FIX(0.195090322) +#define COS_1_16 FIX(0.980785280) #define SIN_7_16 COS_1_16 #define COS_7_16 SIN_1_16 -#define SIN_3_16 FIX(0.5555702330196022) -#define COS_3_16 FIX(0.8314696123025450) +#define SIN_3_16 FIX(0.555570233) +#define COS_3_16 FIX(0.831469612) #define SIN_5_16 COS_3_16 #define COS_5_16 SIN_3_16 -#define OSIN_1_4 FIXO(0.707106781185647) +/* OSIN_i_j is sine of i*pi/j, scaled by DCT_SCALE/OVERSCALE */ +/* OCOS_i_j is cosine of i*pi/j, scaled by DCT_SCALE/OVERSCALE */ + +#define OSIN_1_4 FIXO(0.707106781) #define OCOS_1_4 OSIN_1_4 -#define OSIN_1_8 FIXO(0.3826834323650898) -#define OCOS_1_8 FIXO(0.9238795325112870) +#define OSIN_1_8 FIXO(0.382683432) +#define OCOS_1_8 FIXO(0.923879533) #define OSIN_3_8 OCOS_1_8 #define OCOS_3_8 OSIN_1_8 -#define OSIN_1_16 FIXO(0.1950903220161282) -#define OCOS_1_16 FIXO(0.9807852804032300) +#define OSIN_1_16 FIXO(0.195090322) +#define OCOS_1_16 FIXO(0.980785280) #define OSIN_7_16 OCOS_1_16 #define OCOS_7_16 OSIN_1_16 -#define OSIN_3_16 FIXO(0.5555702330196022) -#define OCOS_3_16 FIXO(0.8314696123025450) +#define OSIN_3_16 FIXO(0.555570233) +#define OCOS_3_16 FIXO(0.831469612) #define OSIN_5_16 OCOS_3_16 #define OCOS_5_16 OSIN_3_16 +/* + * Perform a 1-dimensional inverse DCT. + * Note that this code is specialized to the case DCTSIZE = 8. + */ + INLINE LOCAL void fast_idct_8 (DCTELEM *in, int stride) { - /* tmp1x are new values of tmpx -- flashy register colourers + /* many tmps have nonoverlapping lifetime -- flashy register colourers * should be able to do this lot very well */ + INT32 in0, in1, in2, in3, in4, in5, in6, in7; INT32 tmp10, tmp11, tmp12, tmp13; INT32 tmp20, tmp21, tmp22, tmp23; INT32 tmp30, tmp31; INT32 tmp40, tmp41, tmp42, tmp43; INT32 tmp50, tmp51, tmp52, tmp53; - INT32 in0, in1, in2, in3, in4, in5, in6, in7; - + SHIFT_TEMPS + in0 = in[ 0]; in1 = in[stride ]; in2 = in[stride*2]; @@ -117,7 +165,9 @@ fast_idct_8 (DCTELEM *in, int stride) in5 = in[stride*5]; in6 = in[stride*6]; in7 = in[stride*7]; - + + /* These values are scaled by DCT_SCALE */ + tmp10 = (in0 + in4) * COS_1_4; tmp11 = (in0 - in4) * COS_1_4; tmp12 = in2 * SIN_1_8 - in6 * COS_1_8; @@ -127,35 +177,43 @@ fast_idct_8 (DCTELEM *in, int stride) tmp21 = tmp11 + tmp12; tmp22 = tmp11 - tmp12; tmp23 = tmp10 - tmp13; - + + /* These values are scaled by OVERSCALE */ + tmp30 = UNFIXO((in3 + in5) * COS_1_4); tmp31 = UNFIXO((in3 - in5) * COS_1_4); - - tmp40 = OVERSH(in1) + tmp30; - tmp41 = OVERSH(in7) + tmp31; - tmp42 = OVERSH(in1) - tmp30; - tmp43 = OVERSH(in7) - tmp31; - + + OVERSHIFT(in1); + OVERSHIFT(in7); + + tmp40 = in1 + tmp30; + tmp41 = in7 + tmp31; + tmp42 = in1 - tmp30; + tmp43 = in7 - tmp31; + + /* And these are scaled by DCT_SCALE */ + tmp50 = tmp40 * OCOS_1_16 + tmp41 * OSIN_1_16; tmp51 = tmp40 * OSIN_1_16 - tmp41 * OCOS_1_16; tmp52 = tmp42 * OCOS_5_16 + tmp43 * OSIN_5_16; tmp53 = tmp42 * OSIN_5_16 - tmp43 * OCOS_5_16; - in[ 0] = UNFIXH(tmp20 + tmp50); - in[stride ] = UNFIXH(tmp21 + tmp53); - in[stride*2] = UNFIXH(tmp22 + tmp52); - in[stride*3] = UNFIXH(tmp23 + tmp51); - in[stride*4] = UNFIXH(tmp23 - tmp51); - in[stride*5] = UNFIXH(tmp22 - tmp52); - in[stride*6] = UNFIXH(tmp21 - tmp53); - in[stride*7] = UNFIXH(tmp20 - tmp50); + in[ 0] = (DCTELEM) UNFIXH(tmp20 + tmp50); + in[stride ] = (DCTELEM) UNFIXH(tmp21 + tmp53); + in[stride*2] = (DCTELEM) UNFIXH(tmp22 + tmp52); + in[stride*3] = (DCTELEM) UNFIXH(tmp23 + tmp51); + in[stride*4] = (DCTELEM) UNFIXH(tmp23 - tmp51); + in[stride*5] = (DCTELEM) UNFIXH(tmp22 - tmp52); + in[stride*6] = (DCTELEM) UNFIXH(tmp21 - tmp53); + in[stride*7] = (DCTELEM) UNFIXH(tmp20 - tmp50); } /* * Perform the inverse DCT on one block of coefficients. * - * Note that this code is specialized to the case DCTSIZE = 8. + * A 2-D IDCT can be done by 1-D IDCT on each row + * followed by 1-D IDCT on each column. */ GLOBAL void diff --git a/jversion.h b/jversion.h new file mode 100644 index 000000000..a93138544 --- /dev/null +++ b/jversion.h @@ -0,0 +1,14 @@ +/* + * jversion.h + * + * Copyright (C) 1991, Thomas G. Lane. + * This file is part of the Independent JPEG Group's software. + * For conditions of distribution and use, see the accompanying README file. + * + * This file contains software version identification. + */ + + +#define JVERSION "2 13-Dec-91" + +#define JCOPYRIGHT "Copyright (C) 1991, Thomas G. Lane" diff --git a/jvirtmem.c b/jvirtmem.c index 4a0627cad..6e1d332c5 100644 --- a/jvirtmem.c +++ b/jvirtmem.c @@ -17,9 +17,11 @@ * They should exit to error_exit if unsuccessful. */ +#define AM_MEMORY_MANAGER /* we define big_Xarray_control structs */ + #include "jinclude.h" -#ifdef __STDC__ +#ifdef INCLUDES_ARE_ANSI #include /* to declare malloc(), free() */ #else extern void * malloc PP((size_t size)); @@ -52,6 +54,20 @@ extern void free PP((void *ptr)); #endif /* NEED_FAR_POINTERS */ +/* + * When allocating 2-D arrays we can either ask malloc() for each row + * individually, or grab the whole space in one chunk. The latter is + * a lot faster on large arrays, but fails if malloc can't handle big + * requests, as is typically true on MS-DOS. + * We assume here that big malloc requests are safe whenever + * NEED_FAR_POINTERS is not defined, but you can change this if you are + * on a weird machine. + */ + +#ifndef NEED_FAR_POINTERS +#define BIG_MALLOCS_OK /* safe to ask far_malloc for > 64Kb */ +#endif + /* * Some important notes: @@ -220,12 +236,20 @@ alloc_small_sarray (long samplesperrow, long numrows) /* Allocate a "small" (all-in-memory) 2-D sample array */ { JSAMPARRAY result; +#ifdef BIG_MALLOCS_OK + JSAMPROW workspace; +#endif long i; #ifdef MEM_STATS total_num_sarray++; +#ifdef BIG_MALLOCS_OK + total_bytes_sarray += numrows * samplesperrow * SIZEOF(JSAMPLE) + + MALLOC_FAR_OVERHEAD; +#else total_bytes_sarray += (samplesperrow * SIZEOF(JSAMPLE) + MALLOC_FAR_OVERHEAD) * numrows; +#endif cur_num_sarray++; if (cur_num_sarray > max_num_sarray) max_num_sarray = cur_num_sarray; #endif @@ -234,11 +258,24 @@ alloc_small_sarray (long samplesperrow, long numrows) result = (JSAMPARRAY) alloc_small((size_t) (numrows * SIZEOF(JSAMPROW))); /* Get the rows themselves; on 80x86 these are "far" */ + +#ifdef BIG_MALLOCS_OK + workspace = (JSAMPROW) far_malloc((size_t) + (numrows * samplesperrow * SIZEOF(JSAMPLE))); + if (workspace == NULL) + out_of_memory(3); for (i = 0; i < numrows; i++) { - result[i] = (JSAMPROW) far_malloc((size_t) (samplesperrow * SIZEOF(JSAMPLE))); + result[i] = workspace; + workspace += samplesperrow; + } +#else + for (i = 0; i < numrows; i++) { + result[i] = (JSAMPROW) far_malloc((size_t) + (samplesperrow * SIZEOF(JSAMPLE))); if (result[i] == NULL) out_of_memory(3); } +#endif return result; } @@ -248,12 +285,16 @@ METHODDEF void free_small_sarray (JSAMPARRAY ptr, long numrows) /* Free a "small" (all-in-memory) 2-D sample array */ { + /* Free the rows themselves; on 80x86 these are "far" */ +#ifdef BIG_MALLOCS_OK + far_free((void FAR *) ptr[0]); +#else long i; - /* Free the rows themselves; on 80x86 these are "far" */ for (i = 0; i < numrows; i++) { far_free((void FAR *) ptr[i]); } +#endif /* Free space for row pointers; this is always "near" on 80x86 */ free_small((void *) ptr); @@ -269,12 +310,20 @@ alloc_small_barray (long blocksperrow, long numrows) /* Allocate a "small" (all-in-memory) 2-D coefficient-block array */ { JBLOCKARRAY result; +#ifdef BIG_MALLOCS_OK + JBLOCKROW workspace; +#endif long i; #ifdef MEM_STATS total_num_barray++; +#ifdef BIG_MALLOCS_OK + total_bytes_barray += numrows * blocksperrow * SIZEOF(JBLOCK) + + MALLOC_FAR_OVERHEAD; +#else total_bytes_barray += (blocksperrow * SIZEOF(JBLOCK) + MALLOC_FAR_OVERHEAD) * numrows; +#endif cur_num_barray++; if (cur_num_barray > max_num_barray) max_num_barray = cur_num_barray; #endif @@ -283,11 +332,24 @@ alloc_small_barray (long blocksperrow, long numrows) result = (JBLOCKARRAY) alloc_small((size_t) (numrows * SIZEOF(JBLOCKROW))); /* Get the rows themselves; on 80x86 these are "far" */ + +#ifdef BIG_MALLOCS_OK + workspace = (JBLOCKROW) far_malloc((size_t) + (numrows * blocksperrow * SIZEOF(JBLOCK))); + if (workspace == NULL) + out_of_memory(4); for (i = 0; i < numrows; i++) { - result[i] = (JBLOCKROW) far_malloc((size_t) (blocksperrow * SIZEOF(JBLOCK))); + result[i] = workspace; + workspace += blocksperrow; + } +#else + for (i = 0; i < numrows; i++) { + result[i] = (JBLOCKROW) far_malloc((size_t) + (blocksperrow * SIZEOF(JBLOCK))); if (result[i] == NULL) out_of_memory(4); } +#endif return result; } @@ -297,12 +359,16 @@ METHODDEF void free_small_barray (JBLOCKARRAY ptr, long numrows) /* Free a "small" (all-in-memory) 2-D coefficient-block array */ { + /* Free the rows themselves; on 80x86 these are "far" */ +#ifdef BIG_MALLOCS_OK + far_free((void FAR *) ptr[0]); +#else long i; - /* Free the rows themselves; on 80x86 these are "far" */ for (i = 0; i < numrows; i++) { far_free((void FAR *) ptr[i]); } +#endif /* Free space for row pointers; this is always "near" on 80x86 */ free_small((void *) ptr); diff --git a/jwrgif.c b/jwrgif.c index 44062ec89..9423af2a4 100644 --- a/jwrgif.c +++ b/jwrgif.c @@ -97,8 +97,9 @@ flush_packet (void) /* flush any accumulated data */ { if (bytesinpkt > 0) { /* never write zero-length packet */ - packetbuf[0] = bytesinpkt++; - if (fwrite(packetbuf, 1, bytesinpkt, dcinfo->output_file) != bytesinpkt) + packetbuf[0] = (char) bytesinpkt++; + if (FWRITE(dcinfo->output_file, packetbuf, bytesinpkt) + != (size_t) bytesinpkt) ERREXIT(dcinfo->emethods, "Output file write error"); bytesinpkt = 0; } @@ -106,7 +107,7 @@ flush_packet (void) LOCAL void -char_out (int c) +char_out (char c) /* Add a character to current packet; flush to disk if necessary */ { packetbuf[++bytesinpkt] = c; @@ -133,7 +134,7 @@ output (code_int code) cur_bits += n_bits; while (cur_bits >= 8) { - char_out((int) (cur_accum & 0xFF)); + char_out((char) (cur_accum & 0xFF)); cur_accum >>= 8; cur_bits -= 8; } @@ -225,7 +226,7 @@ compress_byte (int c) i -= HSIZE; if (hash_code[i] != 0) { /* is first probed slot empty? */ - if (hash_prefix[i] == waiting_code && hash_suffix[i] == c) { + if (hash_prefix[i] == waiting_code && hash_suffix[i] == (UINT8) c) { waiting_code = hash_code[i]; return; } @@ -239,7 +240,7 @@ compress_byte (int c) i += HSIZE; if (hash_code[i] == 0) break; /* hit empty slot */ - if (hash_prefix[i] == waiting_code && hash_suffix[i] == c) { + if (hash_prefix[i] == waiting_code && hash_suffix[i] == (UINT8) c) { waiting_code = hash_code[i]; return; } @@ -251,7 +252,7 @@ compress_byte (int c) if (free_code < LZW_TABLE_SIZE) { hash_code[i] = free_code++; /* add symbol to hashtable */ hash_prefix[i] = waiting_code; - hash_suffix[i] = c; + hash_suffix[i] = (UINT8) c; } else clear_block(); waiting_code = c; @@ -269,7 +270,7 @@ compress_term (void) output(EOFCode); /* Flush the bit-packing buffer */ if (cur_bits > 0) { - char_out((int) (cur_accum & 0xFF)); + char_out((char) (cur_accum & 0xFF)); } /* Flush the packet buffer */ flush_packet(); @@ -334,7 +335,7 @@ emit_header (int num_colors, JSAMPARRAY colormap) * Write the GIF header. * Note that we generate a plain GIF87 header for maximum compatibility. */ - fwrite("GIF87a", 1, 6, dcinfo->output_file); + (void) FWRITE(dcinfo->output_file, "GIF87a", 6); /* Write the Logical Screen Descriptor */ put_word((UINT16) dcinfo->image_width); put_word((UINT16) dcinfo->image_height); diff --git a/jwrjfif.c b/jwrjfif.c index cc379ac65..28813554e 100644 --- a/jwrjfif.c +++ b/jwrjfif.c @@ -36,7 +36,8 @@ /* Write some bytes from a (char *) buffer */ #define WRITE_BYTES(cinfo,dataptr,datacount) \ - { if (fwrite((dataptr), 1, (datacount), cinfo->output_file) != (datacount)) \ + { if (FWRITE(cinfo->output_file, dataptr, datacount) \ + != (size_t) (datacount)) \ ERREXIT(cinfo->emethods, "Output file write error"); } /* Clean up and verify successful output */ @@ -247,6 +248,9 @@ emit_sof (compress_info_ptr cinfo, JPEG_MARKER code) emit_2bytes(cinfo, 3 * cinfo->num_components + 2 + 5 + 1); /* length */ + if (cinfo->image_height > 65535L || cinfo->image_width > 65535L) + ERREXIT(cinfo->emethods, "Maximum image dimension for JFIF is 65535 pixels"); + emit_byte(cinfo, cinfo->data_precision); emit_2bytes(cinfo, (int) cinfo->image_height); emit_2bytes(cinfo, (int) cinfo->image_width); diff --git a/jwrppm.c b/jwrppm.c index 41687fec1..7a5255c4b 100644 --- a/jwrppm.c +++ b/jwrppm.c @@ -21,6 +21,16 @@ #ifdef PPM_SUPPORTED +/* + * Haven't yet got around to making this work with text-format output, + * hence cannot handle pixels wider than 8 bits. + */ + +#ifndef EIGHT_BIT_SAMPLES + Sorry, this code only copes with 8-bit JSAMPLEs. /* deliberate syntax err */ +#endif + + static JSAMPARRAY color_map; /* saves color map passed by quantizer */ diff --git a/jwrrle.c b/jwrrle.c new file mode 100644 index 000000000..f324758f7 --- /dev/null +++ b/jwrrle.c @@ -0,0 +1,231 @@ +/* + * jwrrle.c + * + * Copyright (C) 1991, Thomas G. Lane. + * This file is part of the Independent JPEG Group's software. + * For conditions of distribution and use, see the accompanying README file. + * + * This file contains routines to write output images in RLE format. + * The Utah Raster Toolkit library is required (version 3.0). + * + * These routines may need modification for non-Unix environments or + * specialized applications. As they stand, they assume output to + * an ordinary stdio stream. + * + * These routines are invoked via the methods put_pixel_rows, put_color_map, + * and output_init/term. + * + * Based on code contributed by Mike Lijewski. + */ + +#include "jinclude.h" + +#ifdef RLE_SUPPORTED + +/* rle.h is provided by the Utah Raster Toolkit. */ + +#include + + +/* + * output_term assumes that JSAMPLE has the same representation as rle_pixel, + * to wit, "unsigned char". Hence we can't cope with 12- or 16-bit samples. + */ + +#ifndef EIGHT_BIT_SAMPLES + Sorry, this code only copes with 8-bit JSAMPLEs. /* deliberate syntax err */ +#endif + + +/* + * Since RLE stores scanlines bottom-to-top, we have to invert the image + * from JPEG's top-to-bottom order. To do this, we save the outgoing data + * in virtual array(s) during put_pixel_row calls, then actually emit the + * RLE file during output_term. We use one virtual array if the output is + * grayscale or colormapped, more if it is full color. + */ + +#define MAX_CHANS 4 /* allow up to four color components */ +static big_sarray_ptr channels[MAX_CHANS]; /* Virtual arrays for saved data */ + +static long cur_output_row; /* next row# to write to virtual array(s) */ + + +/* + * For now, if we emit an RLE color map then it is always 256 entries long, + * though not all of the entries need be used. + */ + +#define CMAPBITS 8 +#define CMAPLENGTH (1<<(CMAPBITS)) + +static rle_map *output_colormap; /* RLE-style color map, or NULL if none */ +static int number_colors; /* Number of colors actually used */ + + +/* + * Write the file header. + * + * In this module it's easier to wait till output_term to actually write + * anything; here we just request the big arrays we'll need. + */ + +METHODDEF void +output_init (decompress_info_ptr cinfo) +{ + short ci; + + if (cinfo->final_out_comps > MAX_CHANS) + ERREXIT1(cinfo->emethods, "Cannot handle %d output channels for RLE", + cinfo->final_out_comps); + + for (ci = 0; ci < cinfo->final_out_comps; ci++) { + channels[ci] = (*cinfo->emethods->request_big_sarray) + (cinfo->image_width, cinfo->image_height, 1L); + } + + output_colormap = NULL; /* No output colormap as yet */ + number_colors = 0; + cur_output_row = 0; /* Start filling virtual arrays at row 0 */ +} + + +/* + * Write some pixel data. + * + * This routine just saves the data away in virtual arrays. + */ + +METHODDEF void +put_pixel_rows (decompress_info_ptr cinfo, int num_rows, + JSAMPIMAGE pixel_data) +{ + JSAMPROW outputrow[1]; /* a pseudo JSAMPARRAY structure */ + int row; + short ci; + + for (row = 0; row < num_rows; row++) { + for (ci = 0; ci < cinfo->final_out_comps; ci++) { + outputrow[0] = *((*cinfo->emethods->access_big_sarray) + (channels[ci], cur_output_row, TRUE)); + jcopy_sample_rows(pixel_data[ci], row, outputrow, 0, + 1, cinfo->image_width); + } + cur_output_row++; + } +} + + +/* + * Write the color map. + * + * For RLE output we just save the colormap for the output stage. + */ + +METHODDEF void +put_color_map (decompress_info_ptr cinfo, int num_colors, JSAMPARRAY colormap) +{ + size_t cmapsize; + short ci; + int i; + + if (num_colors > CMAPLENGTH) + ERREXIT1(cinfo->emethods, "Cannot handle %d colormap entries for RLE", + num_colors); + + /* Allocate storage for RLE-style cmap, zero any extra entries */ + cmapsize = cinfo->color_out_comps * CMAPLENGTH * SIZEOF(rle_map); + output_colormap = (rle_map *) (*cinfo->emethods->alloc_small) (cmapsize); + MEMZERO((void *) output_colormap, cmapsize); + + /* Save away data in RLE format --- note 8-bit left shift! */ + /* Shifting would need adjustment for JSAMPLEs wider than 8 bits. */ + for (ci = 0; ci < cinfo->color_out_comps; ci++) { + for (i = 0; i < num_colors; i++) { + output_colormap[ci * CMAPLENGTH + i] = GETJSAMPLE(colormap[ci][i]) << 8; + } + } + number_colors = num_colors; +} + + +/* + * Finish up at the end of the file. + * + * Here is where we really output the RLE file. + */ + +METHODDEF void +output_term (decompress_info_ptr cinfo) +{ + rle_hdr header; /* Output file information */ + rle_pixel *output_rows[MAX_CHANS]; + char cmapcomment[80]; + short ci; + long row; + + /* Initialize the header info */ + MEMZERO((void *) &header, SIZEOF(rle_hdr)); /* make sure all bits are 0 */ + header.rle_file = cinfo->output_file; + header.xmin = 0; + header.xmax = cinfo->image_width - 1; + header.ymin = 0; + header.ymax = cinfo->image_height - 1; + header.alpha = 0; + header.ncolors = cinfo->final_out_comps; + for (ci = 0; ci < cinfo->final_out_comps; ci++) { + RLE_SET_BIT(header, ci); + } + if (number_colors > 0) { + header.ncmap = cinfo->color_out_comps; + header.cmaplen = CMAPBITS; + header.cmap = output_colormap; + /* Add a comment to the output image with the true colormap length. */ + sprintf(cmapcomment, "color_map_length=%d", number_colors); + rle_putcom(cmapcomment, &header); + } + /* Emit the RLE header and color map (if any) */ + rle_put_setup(&header); + + /* Now output the RLE data from our virtual array(s). + * We assume here that (a) rle_pixel is represented the same as JSAMPLE, + * and (b) we are not on a machine where FAR pointers differ from regular. + */ + for (row = cinfo->image_height-1; row >= 0; row--) { + for (ci = 0; ci < cinfo->final_out_comps; ci++) { + output_rows[ci] = (rle_pixel *) *((*cinfo->emethods->access_big_sarray) + (channels[ci], row, FALSE)); + } + rle_putrow(output_rows, (int) cinfo->image_width, &header); + } + + /* Emit file trailer */ + rle_puteof(&header); + fflush(cinfo->output_file); + if (ferror(cinfo->output_file)) + ERREXIT(cinfo->emethods, "Output file write error"); + + /* Release memory */ + for (ci = 0; ci < cinfo->final_out_comps; ci++) { + (*cinfo->emethods->free_big_sarray) (channels[ci]); + } + if (output_colormap != NULL) + (*cinfo->emethods->free_small) ((void *) output_colormap); +} + + +/* + * The method selection routine for RLE format output. + * This should be called from d_ui_method_selection if RLE output is wanted. + */ + +GLOBAL void +jselwrle (decompress_info_ptr cinfo) +{ + cinfo->methods->output_init = output_init; + cinfo->methods->put_color_map = put_color_map; + cinfo->methods->put_pixel_rows = put_pixel_rows; + cinfo->methods->output_term = output_term; +} + +#endif /* RLE_SUPPORTED */ diff --git a/jwrtarga.c b/jwrtarga.c new file mode 100644 index 000000000..72259e773 --- /dev/null +++ b/jwrtarga.c @@ -0,0 +1,223 @@ +/* + * jwrtarga.c + * + * Copyright (C) 1991, Thomas G. Lane. + * This file is part of the Independent JPEG Group's software. + * For conditions of distribution and use, see the accompanying README file. + * + * This file contains routines to write output images in Targa format. + * + * These routines may need modification for non-Unix environments or + * specialized applications. As they stand, they assume output to + * an ordinary stdio stream. + * + * These routines are invoked via the methods put_pixel_rows, put_color_map, + * and output_init/term. + * + * Based on code contributed by Lee Daniel Crocker. + */ + +#include "jinclude.h" + +#ifdef TARGA_SUPPORTED + + +/* + * To support 12-bit JPEG data, we'd have to scale output down to 8 bits. + * This is not yet implemented. + */ + +#ifndef EIGHT_BIT_SAMPLES + Sorry, this code only copes with 8-bit JSAMPLEs. /* deliberate syntax err */ +#endif + + +static JSAMPARRAY color_map; /* saves color map passed by quantizer */ + + +LOCAL void +write_header (decompress_info_ptr cinfo, int num_colors) +/* Create and write a Targa header */ +{ + char targaheader[18]; + + /* Set unused fields of header to 0 */ + MEMZERO((void *) targaheader, SIZEOF(targaheader)); + + if (num_colors > 0) { + targaheader[1] = 1; /* color map type 1 */ + targaheader[5] = (char) (num_colors & 0xFF); + targaheader[6] = (char) (num_colors >> 8); + targaheader[7] = 24; /* 24 bits per cmap entry */ + } + + targaheader[12] = (char) (cinfo->image_width & 0xFF); + targaheader[13] = (char) (cinfo->image_width >> 8); + targaheader[14] = (char) (cinfo->image_height & 0xFF); + targaheader[15] = (char) (cinfo->image_height >> 8); + targaheader[17] = 0x20; /* Top-down, non-interlaced */ + + if (cinfo->out_color_space == CS_GRAYSCALE) { + targaheader[2] = 3; /* image type = uncompressed gray-scale */ + targaheader[16] = 8; /* bits per pixel */ + } else { /* must be RGB */ + if (num_colors > 0) { + targaheader[2] = 1; /* image type = colormapped RGB */ + targaheader[16] = 8; + } else { + targaheader[2] = 2; /* image type = uncompressed RGB */ + targaheader[16] = 24; + } + } + + if (FWRITE(cinfo->output_file, targaheader, 18) != (size_t) 18) + ERREXIT(cinfo->emethods, "Could not write Targa header"); +} + + +/* + * Write the file header. + */ + +METHODDEF void +output_init (decompress_info_ptr cinfo) +{ + if (cinfo->out_color_space == CS_GRAYSCALE) { + /* Targa doesn't have a mapped grayscale format, so we will */ + /* demap quantized gray output. Never emit a colormap. */ + write_header(cinfo, 0); + } else if (cinfo->out_color_space == CS_RGB) { + /* For quantized output, defer writing header until put_color_map time. */ + if (! cinfo->quantize_colors) + write_header(cinfo, 0); + } else { + ERREXIT(cinfo->emethods, "Targa output must be grayscale or RGB"); + } +} + + +/* + * Write some pixel data. + */ + +METHODDEF void +put_pixel_rows (decompress_info_ptr cinfo, int num_rows, + JSAMPIMAGE pixel_data) +{ + register FILE * outfile = cinfo->output_file; + register JSAMPROW ptr0, ptr1, ptr2; + register long col; + register long width = cinfo->image_width; + register int row; + + if (cinfo->final_out_comps == 1) { + /* here for grayscale or quantized color output */ + for (row = 0; row < num_rows; row++) { + ptr0 = pixel_data[0][row]; + for (col = width; col > 0; col--) { + putc(GETJSAMPLE(*ptr0), outfile); + ptr0++; + } + } + } else { + /* here for unquantized color output */ + for (row = 0; row < num_rows; row++) { + ptr0 = pixel_data[0][row]; + ptr1 = pixel_data[1][row]; + ptr2 = pixel_data[2][row]; + for (col = width; col > 0; col--) { + putc(GETJSAMPLE(*ptr2), outfile); /* write in BGR order */ + ptr2++; + putc(GETJSAMPLE(*ptr1), outfile); + ptr1++; + putc(GETJSAMPLE(*ptr0), outfile); + ptr0++; + } + } + } +} + + +/* + * Write some demapped pixel data when color quantization is in effect. + * For Targa, this is only applied to grayscale data. + */ + +METHODDEF void +put_demapped_rows (decompress_info_ptr cinfo, int num_rows, + JSAMPIMAGE pixel_data) +{ + register FILE * outfile = cinfo->output_file; + register JSAMPROW ptr; + register long col; + register long width = cinfo->image_width; + register int row; + + for (row = 0; row < num_rows; row++) { + ptr = pixel_data[0][row]; + for (col = width; col > 0; col--) { + putc(GETJSAMPLE(color_map[0][GETJSAMPLE(*ptr)]), outfile); + ptr++; + } + } +} + + +/* + * Write the color map. + */ + +METHODDEF void +put_color_map (decompress_info_ptr cinfo, int num_colors, JSAMPARRAY colormap) +{ + register FILE * outfile = cinfo->output_file; + int i; + + if (cinfo->out_color_space == CS_RGB) { + /* We only support 8-bit colormap indexes, so only 256 colors */ + if (num_colors > 256) + ERREXIT(cinfo->emethods, "Too many colors for Targa output"); + /* Time to write the header */ + write_header(cinfo, num_colors); + /* Write the colormap. Note Targa uses BGR byte order */ + for (i = 0; i < num_colors; i++) { + putc(GETJSAMPLE(colormap[2][i]), outfile); + putc(GETJSAMPLE(colormap[1][i]), outfile); + putc(GETJSAMPLE(colormap[0][i]), outfile); + } + } else { + color_map = colormap; /* save for use in output */ + cinfo->methods->put_pixel_rows = put_demapped_rows; + } +} + + +/* + * Finish up at the end of the file. + */ + +METHODDEF void +output_term (decompress_info_ptr cinfo) +{ + /* No work except to make sure we wrote the output file OK */ + fflush(cinfo->output_file); + if (ferror(cinfo->output_file)) + ERREXIT(cinfo->emethods, "Output file write error"); +} + + +/* + * The method selection routine for Targa format output. + * This should be called from d_ui_method_selection if Targa output is wanted. + */ + +GLOBAL void +jselwtarga (decompress_info_ptr cinfo) +{ + cinfo->methods->output_init = output_init; + cinfo->methods->put_color_map = put_color_map; + cinfo->methods->put_pixel_rows = put_pixel_rows; + cinfo->methods->output_term = output_term; +} + +#endif /* TARGA_SUPPORTED */ diff --git a/makcjpeg.cf b/makcjpeg.cf index ea93ce4bd..7c9d736df 100644 --- a/makcjpeg.cf +++ b/makcjpeg.cf @@ -1,5 +1,6 @@ -L jcmain.mix jerror.mix jcdeflts.mix jrdgif.mix jrdppm.mix jwrjfif.mix -L jcarith.mix jccolor.mix jcexpand.mix jchuff.mix jcmaster.mix jcmcu.mix -L jcpipe.mix jcsample.mix jfwddct.mix jutils.mix jvirtmem.mix +L jcmain.mix jcmaster.mix jcdeflts.mix jcarith.mix jccolor.mix jcexpand.mix +L jchuff.mix jcmcu.mix jcpipe.mix jcsample.mix jfwddct.mix jwrjfif.mix +L jrdgif.mix jrdppm.mix jrdrle.mix jrdtarga.mix jutils.mix jvirtmem.mix +L jerror.mix fa; b cjpeg,8K,48K, diff --git a/makcjpeg.lnk b/makcjpeg.lnk index 23702b22b..258f644bd 100644 --- a/makcjpeg.lnk +++ b/makcjpeg.lnk @@ -1,20 +1,22 @@ jcmain.obj + -jerror.obj + +jcmaster.obj + jcdeflts.obj + -jrdgif.obj + -jrdppm.obj + -jwrjfif.obj + jcarith.obj + jccolor.obj + jcexpand.obj + jchuff.obj + -jcmaster.obj + jcmcu.obj + jcpipe.obj + jcsample.obj + jfwddct.obj + +jwrjfif.obj + +jrdgif.obj + +jrdppm.obj + +jrdrle.obj + +jrdtarga.obj + jutils.obj + -jvirtmem.obj +jvirtmem.obj + +jerror.obj cjpeg.exe /NOI nul.map diff --git a/makcjpeg.lst b/makcjpeg.lst new file mode 100644 index 000000000..aa758b363 --- /dev/null +++ b/makcjpeg.lst @@ -0,0 +1,4 @@ +jcmain.obj jcmaster.obj jcdeflts.obj jcarith.obj jccolor.obj jcexpand.obj +jchuff.obj jcmcu.obj jcpipe.obj jcsample.obj jfwddct.obj jwrjfif.obj +jrdgif.obj jrdppm.obj jrdrle.obj jrdtarga.obj jutils.obj jvirtmem.obj +jerror.obj diff --git a/makdjpeg.cf b/makdjpeg.cf index 968af311f..ea02876ba 100644 --- a/makdjpeg.cf +++ b/makdjpeg.cf @@ -1,6 +1,6 @@ -L jdmain.mix jerror.mix jrdjfif.mix jwrgif.mix jwrppm.mix -L jbsmooth.mix jdarith.mix jdcolor.mix jdhuff.mix jdmaster.mix jdmcu.mix -L jdpipe.mix jdsample.mix jquant1.mix jquant2.mix jrevdct.mix jutils.mix -L jvirtmem.mix +L jdmain.mix jdmaster.mix jddeflts.mix jbsmooth.mix jdarith.mix jdcolor.mix +L jdhuff.mix jdmcu.mix jdpipe.mix jdsample.mix jquant1.mix jquant2.mix +L jrevdct.mix jrdjfif.mix jwrgif.mix jwrppm.mix jwrrle.mix jwrtarga.mix +L jutils.mix jvirtmem.mix jerror.mix fa; b djpeg,8K,48K, diff --git a/makdjpeg.lnk b/makdjpeg.lnk index 6e9107cf9..7f909db06 100644 --- a/makdjpeg.lnk +++ b/makdjpeg.lnk @@ -1,22 +1,25 @@ jdmain.obj + -jerror.obj + -jrdjfif.obj + -jwrgif.obj + -jwrppm.obj + +jdmaster.obj + +jddeflts.obj + jbsmooth.obj + jdarith.obj + jdcolor.obj + jdhuff.obj + -jdmaster.obj + jdmcu.obj + jdpipe.obj + jdsample.obj + jquant1.obj + jquant2.obj + jrevdct.obj + +jrdjfif.obj + +jwrgif.obj + +jwrppm.obj + +jwrrle.obj + +jwrtarga.obj + jutils.obj + -jvirtmem.obj /NOI -djpeg.exe +jvirtmem.obj + +jerror.obj +djpeg.exe /NOI nul.map nul.def diff --git a/makdjpeg.lst b/makdjpeg.lst new file mode 100644 index 000000000..9c9e72494 --- /dev/null +++ b/makdjpeg.lst @@ -0,0 +1,4 @@ +jdmain.obj jdmaster.obj jddeflts.obj jbsmooth.obj jdarith.obj jdcolor.obj +jdhuff.obj jdmcu.obj jdpipe.obj jdsample.obj jquant1.obj jquant2.obj +jrevdct.obj jrdjfif.obj jwrgif.obj jwrppm.obj jwrrle.obj jwrtarga.obj +jutils.obj jvirtmem.obj jerror.obj diff --git a/makefile.amiga b/makefile.amiga deleted file mode 100644 index 2fde93abf..000000000 --- a/makefile.amiga +++ /dev/null @@ -1,143 +0,0 @@ -# Makefile for Independent JPEG Group's software - -# This makefile is suitable for Amiga systems using Manx Aztec C ver 5.x. -# Thanks to D.J. James for this version. - -# See README and edit jconfig.h before saying "make" !! - -CC= cc - -# You may need to adjust these cc options: -CFLAGS= -MC -MD -DTWO_FILE_COMMANDLINE -LDFLAGS= -LDLIBS= -lml -lcl - -# miscellaneous OS-dependent stuff -LN= ln # linker -RM= delete quiet # file deletion command -AR= lb # library (.lib) file creation command - - -# source files -INCLUDES= jinclude.h jconfig.h jpegdata.h -SOURCES= jbsmooth.c jcarith.c jccolor.c jcdeflts.c jcexpand.c \ - jchuff.c jcmain.c jcmaster.c jcmcu.c jcpipe.c jcsample.c \ - jdarith.c jdcolor.c jdhuff.c jdmain.c jdmaster.c jdmcu.c \ - jdpipe.c jdsample.c jerror.c jfwddct.c jquant1.c jquant2.c \ - jrdjfif.c jrdgif.c jrdppm.c jrevdct.c jutils.c jvirtmem.c \ - jwrjfif.c jwrgif.c jwrppm.c egetopt.c -DOCS= README architecture codingrules -MAKEFILES= makefile.unix makefile.amiga \ - makefile.mc5 makefile.mc6 makcjpeg.lnk makdjpeg.lnk \ - makefile.pwc makcjpeg.cf makdjpeg.cf makljpeg.cf -TESTFILES= testorig.jpg testimg.ppm testimg.jpg -DISTFILES= $(DOCS) $(MAKEFILES) ansi2knr.c $(SOURCES) $(INCLUDES) $(TESTFILES) - -# compression objectfiles -COBJECTS = jcmain.o jcmaster.o jcdeflts.o jcarith.o jccolor.o jcexpand.o \ - jchuff.o jcmcu.o jcpipe.o jcsample.o jfwddct.o \ - jrdgif.o jrdppm.o jwrjfif.o \ - jutils.o jvirtmem.o jerror.o -# decompression objectfiles -DOBJECTS = jdmain.o jdmaster.o jbsmooth.o jdarith.o jdcolor.o jdhuff.o \ - jdmcu.o jdpipe.o jdsample.o jquant1.o jquant2.o jrevdct.o \ - jrdjfif.o jwrgif.o jwrppm.o \ - jutils.o jvirtmem.o jerror.o -# These objectfiles are included in libjpeg.lib (all but jcmain.o, jdmain.o) -LIBOBJECTS = jcmaster.o jcdeflts.o jcarith.o jccolor.o jcexpand.o \ - jchuff.o jcmcu.o jcpipe.o jcsample.o jfwddct.o \ - jrdgif.o jrdppm.o jwrjfif.o \ - jdmaster.o jbsmooth.o jdarith.o jdcolor.o jdhuff.o \ - jdmcu.o jdpipe.o jdsample.o jquant1.o jquant2.o jrevdct.o \ - jrdjfif.o jwrgif.o jwrppm.o \ - jutils.o jvirtmem.o jerror.o - - -all: cjpeg djpeg -# By default, libjpeg.lib is not built unless you explicitly request it. - - -# If you have a C compiler that doesn't understand function prototypes, -# uncomment the 5 lines below and make sure PROTO is not defined by jconfig.h. -# Then say "make ansi2knr" before "make". - -#.c.o: -# ./ansi2knr $*.c tmpansi.c -# $(CC) $(CFLAGS) -c tmpansi.c -# mv tmpansi.o $*.o -# $(RM) tmpansi.c - -ansi2knr: ansi2knr.c - $(CC) -o ansi2knr ansi2knr.c -# You may need to add one of -DBSD, -DVMS, or -DMSDOS to the line above. - - -cjpeg: $(COBJECTS) - $(LN) $(LDFLAGS) -o cjpeg $(COBJECTS) $(LDLIBS) - -djpeg: $(DOBJECTS) - $(LN) $(LDFLAGS) -o djpeg $(DOBJECTS) $(LDLIBS) - -# libjpeg.lib is useful if you are including the JPEG software in a larger -# program; you'd include it in your link, rather than the individual modules. -libjpeg.lib: $(LIBOBJECTS) - -$(RM) libjpeg.lib - $(AR) libjpeg.lib $(LIBOBJECTS) - -# Use the following to test the built library -#cjpeg: jcmain.o libjpeg.lib -# $(LN) $(LDFLAGS) -o cjpeg jcmain.o -llibjpeg $(LDLIBS) -# -#djpeg: jdmain.o libjpeg.lib -# $(LN) $(LDFLAGS) -o djpeg jdmain.o -llibjpeg $(LDLIBS) - -clean: - -$(RM) *.o cjpeg djpeg libjpeg.lib ansi2knr core tmpansi.* testout.ppm testout.jpg - -distribute: - -$(RM) jpegsrc.tar* - tar cvf jpegsrc.tar $(DISTFILES) - list jpegsrc.tar - compress -v jpegsrc.tar - list jpegsrc.tar* - -test: cjpeg djpeg - -$(RM) testout.ppm testout.jpg - ./djpeg testorig.jpg testout.ppm - ./cjpeg testimg.ppm testout.jpg - cmp testimg.ppm testout.ppm - cmp testimg.jpg testout.jpg - - -jbsmooth.o : jbsmooth.c jinclude.h jconfig.h jpegdata.h -jcarith.o : jcarith.c jinclude.h jconfig.h jpegdata.h -jccolor.o : jccolor.c jinclude.h jconfig.h jpegdata.h -jcdeflts.o : jcdeflts.c jinclude.h jconfig.h jpegdata.h -jcexpand.o : jcexpand.c jinclude.h jconfig.h jpegdata.h -jchuff.o : jchuff.c jinclude.h jconfig.h jpegdata.h -jcmain.o : jcmain.c jinclude.h jconfig.h jpegdata.h egetopt.c -jcmaster.o : jcmaster.c jinclude.h jconfig.h jpegdata.h -jcmcu.o : jcmcu.c jinclude.h jconfig.h jpegdata.h -jcpipe.o : jcpipe.c jinclude.h jconfig.h jpegdata.h -jcsample.o : jcsample.c jinclude.h jconfig.h jpegdata.h -jdarith.o : jdarith.c jinclude.h jconfig.h jpegdata.h -jdcolor.o : jdcolor.c jinclude.h jconfig.h jpegdata.h -jdhuff.o : jdhuff.c jinclude.h jconfig.h jpegdata.h -jdmain.o : jdmain.c jinclude.h jconfig.h jpegdata.h egetopt.c -jdmaster.o : jdmaster.c jinclude.h jconfig.h jpegdata.h -jdmcu.o : jdmcu.c jinclude.h jconfig.h jpegdata.h -jdpipe.o : jdpipe.c jinclude.h jconfig.h jpegdata.h -jdsample.o : jdsample.c jinclude.h jconfig.h jpegdata.h -jerror.o : jerror.c jinclude.h jconfig.h jpegdata.h -jfwddct.o : jfwddct.c jinclude.h jconfig.h jpegdata.h -jquant1.o : jquant1.c jinclude.h jconfig.h jpegdata.h -jquant2.o : jquant2.c jinclude.h jconfig.h jpegdata.h -jrdjfif.o : jrdjfif.c jinclude.h jconfig.h jpegdata.h -jrdgif.o : jrdgif.c jinclude.h jconfig.h jpegdata.h -jrdppm.o : jrdppm.c jinclude.h jconfig.h jpegdata.h -jrevdct.o : jrevdct.c jinclude.h jconfig.h jpegdata.h -jutils.o : jutils.c jinclude.h jconfig.h jpegdata.h -jvirtmem.o : jvirtmem.c jinclude.h jconfig.h jpegdata.h -jwrjfif.o : jwrjfif.c jinclude.h jconfig.h jpegdata.h -jwrgif.o : jwrgif.c jinclude.h jconfig.h jpegdata.h -jwrppm.o : jwrppm.c jinclude.h jconfig.h jpegdata.h diff --git a/makefile.ansi b/makefile.ansi new file mode 100644 index 000000000..4b99f083c --- /dev/null +++ b/makefile.ansi @@ -0,0 +1,140 @@ +# Makefile for Independent JPEG Group's software + +# This makefile is suitable for Unix-like systems with ANSI-capable compilers. +# If you have a non-ANSI compiler, makefile.unix is a better starting point. + +# Read SETUP instructions before saying "make" !! + +# The name of your C compiler: +CC= cc + +# You may need to adjust these cc options: +CFLAGS= -O +# In particular: +# Add -DBSD if on a pure BSD system (see jinclude.h). +# Add -DMEM_STATS to enable gathering of memory usage statistics. +# You may also want to add -DTWO_FILE_COMMANDLINE or -D switches for other +# symbols listed in jconfig.h, if you prefer not to change jconfig.h. + +# Link-time cc options: +LDFLAGS= + +# To link any special libraries, add the necessary -l commands here. +# In particular, on some versions of HP-UX (and probably other SysV-derived +# systems) there is a faster alternate malloc(3) library that you can use +# by adding "-lmalloc" to this line. +LDLIBS= + +# miscellaneous OS-dependent stuff +LN= $(CC) # linker +RM= rm -f # file deletion command +AR= ar rc # library (.a) file creation command +AR2= ranlib # second step in .a creation (use "touch" if not needed) + + +# source files (independently compilable files) +SOURCES= jbsmooth.c jcarith.c jccolor.c jcdeflts.c jcexpand.c jchuff.c \ + jcmain.c jcmaster.c jcmcu.c jcpipe.c jcsample.c jdarith.c jdcolor.c \ + jddeflts.c jdhuff.c jdmain.c jdmaster.c jdmcu.c jdpipe.c jdsample.c \ + jerror.c jquant1.c jquant2.c jfwddct.c jrevdct.c jutils.c \ + jvirtmem.c jrdjfif.c jrdgif.c jrdppm.c jrdrle.c jrdtarga.c \ + jwrjfif.c jwrgif.c jwrppm.c jwrrle.c jwrtarga.c +# files included by source files +INCLUDES= jinclude.h jconfig.h jpegdata.h jversion.h egetopt.c +# documentation, test, and support files +DOCS= README SETUP USAGE CHANGELOG cjpeg.1 djpeg.1 architecture codingrules +MAKEFILES= makefile.ansi makefile.unix makefile.manx makefile.sas \ + makefile.mc5 makefile.mc6 makcjpeg.lnk makdjpeg.lnk makefile.tc \ + makcjpeg.lst makdjpeg.lst makefile.pwc makcjpeg.cf makdjpeg.cf \ + makljpeg.cf +OTHERFILES= ansi2knr.c config.c +TESTFILES= testorig.jpg testimg.ppm testimg.jpg +DISTFILES= $(DOCS) $(MAKEFILES) $(SOURCES) $(INCLUDES) $(OTHERFILES) \ + $(TESTFILES) +# objectfiles common to cjpeg and djpeg +COMOBJECTS= jutils.o jvirtmem.o jerror.o +# compression objectfiles +CLIBOBJECTS= jcmaster.o jcdeflts.o jcarith.o jccolor.o jcexpand.o jchuff.o \ + jcmcu.o jcpipe.o jcsample.o jfwddct.o jwrjfif.o jrdgif.o jrdppm.o \ + jrdrle.o jrdtarga.o +COBJECTS= jcmain.o $(CLIBOBJECTS) $(COMOBJECTS) +# decompression objectfiles +DLIBOBJECTS= jdmaster.o jddeflts.o jbsmooth.o jdarith.o jdcolor.o jdhuff.o \ + jdmcu.o jdpipe.o jdsample.o jquant1.o jquant2.o jrevdct.o jrdjfif.o \ + jwrgif.o jwrppm.o jwrrle.o jwrtarga.o +DOBJECTS= jdmain.o $(DLIBOBJECTS) $(COMOBJECTS) +# These objectfiles are included in libjpeg.a +LIBOBJECTS= $(CLIBOBJECTS) $(DLIBOBJECTS) $(COMOBJECTS) + + +all: cjpeg djpeg +# By default, libjpeg.a is not built unless you explicitly request it. +# You can add libjpeg.a to the line above if you want it built by default. + + +cjpeg: $(COBJECTS) + $(LN) $(LDFLAGS) -o cjpeg $(COBJECTS) $(LDLIBS) + +djpeg: $(DOBJECTS) + $(LN) $(LDFLAGS) -o djpeg $(DOBJECTS) $(LDLIBS) + +# libjpeg.a is useful if you are including the JPEG software in a larger +# program; you'd include it in your link, rather than the individual modules. +libjpeg.a: $(LIBOBJECTS) + $(RM) libjpeg.a + $(AR) libjpeg.a $(LIBOBJECTS) + $(AR2) libjpeg.a + +clean: + $(RM) *.o cjpeg djpeg libjpeg.a core testout.ppm testout.jpg + +distribute: + $(RM) jpegsrc.tar* + tar cvf jpegsrc.tar $(DISTFILES) + compress -v jpegsrc.tar + +test: cjpeg djpeg + $(RM) testout.ppm testout.jpg + ./djpeg testorig.jpg >testout.ppm + ./cjpeg testimg.ppm >testout.jpg + cmp testimg.ppm testout.ppm + cmp testimg.jpg testout.jpg + + +jbsmooth.o : jbsmooth.c jinclude.h jconfig.h jpegdata.h +jcarith.o : jcarith.c jinclude.h jconfig.h jpegdata.h +jccolor.o : jccolor.c jinclude.h jconfig.h jpegdata.h +jcdeflts.o : jcdeflts.c jinclude.h jconfig.h jpegdata.h +jcexpand.o : jcexpand.c jinclude.h jconfig.h jpegdata.h +jchuff.o : jchuff.c jinclude.h jconfig.h jpegdata.h +jcmain.o : jcmain.c jinclude.h jconfig.h jpegdata.h jversion.h egetopt.c +jcmaster.o : jcmaster.c jinclude.h jconfig.h jpegdata.h +jcmcu.o : jcmcu.c jinclude.h jconfig.h jpegdata.h +jcpipe.o : jcpipe.c jinclude.h jconfig.h jpegdata.h +jcsample.o : jcsample.c jinclude.h jconfig.h jpegdata.h +jdarith.o : jdarith.c jinclude.h jconfig.h jpegdata.h +jdcolor.o : jdcolor.c jinclude.h jconfig.h jpegdata.h +jddeflts.o : jddeflts.c jinclude.h jconfig.h jpegdata.h +jdhuff.o : jdhuff.c jinclude.h jconfig.h jpegdata.h +jdmain.o : jdmain.c jinclude.h jconfig.h jpegdata.h jversion.h egetopt.c +jdmaster.o : jdmaster.c jinclude.h jconfig.h jpegdata.h +jdmcu.o : jdmcu.c jinclude.h jconfig.h jpegdata.h +jdpipe.o : jdpipe.c jinclude.h jconfig.h jpegdata.h +jdsample.o : jdsample.c jinclude.h jconfig.h jpegdata.h +jerror.o : jerror.c jinclude.h jconfig.h jpegdata.h +jquant1.o : jquant1.c jinclude.h jconfig.h jpegdata.h +jquant2.o : jquant2.c jinclude.h jconfig.h jpegdata.h +jfwddct.o : jfwddct.c jinclude.h jconfig.h jpegdata.h +jrevdct.o : jrevdct.c jinclude.h jconfig.h jpegdata.h +jutils.o : jutils.c jinclude.h jconfig.h jpegdata.h +jvirtmem.o : jvirtmem.c jinclude.h jconfig.h jpegdata.h +jrdjfif.o : jrdjfif.c jinclude.h jconfig.h jpegdata.h +jrdgif.o : jrdgif.c jinclude.h jconfig.h jpegdata.h +jrdppm.o : jrdppm.c jinclude.h jconfig.h jpegdata.h +jrdrle.o : jrdrle.c jinclude.h jconfig.h jpegdata.h +jrdtarga.o : jrdtarga.c jinclude.h jconfig.h jpegdata.h +jwrjfif.o : jwrjfif.c jinclude.h jconfig.h jpegdata.h +jwrgif.o : jwrgif.c jinclude.h jconfig.h jpegdata.h +jwrppm.o : jwrppm.c jinclude.h jconfig.h jpegdata.h +jwrrle.o : jwrrle.c jinclude.h jconfig.h jpegdata.h +jwrtarga.o : jwrtarga.c jinclude.h jconfig.h jpegdata.h diff --git a/makefile.manx b/makefile.manx new file mode 100644 index 000000000..bb89926e9 --- /dev/null +++ b/makefile.manx @@ -0,0 +1,133 @@ +# Makefile for Independent JPEG Group's software + +# This makefile is for Amiga systems using Manx Aztec C ver 5.x. +# Thanks to D.J. James for this version. + +# Read SETUP instructions before saying "make" !! + +# The name of your C compiler: +CC= cc + +# You may need to adjust these cc options: +CFLAGS= -MC -MD -DTWO_FILE_COMMANDLINE + +# Link-time cc options: +LDFLAGS= + +# To link any special libraries, add the necessary -l commands here. +LDLIBS= -lml -lcl + +# miscellaneous OS-dependent stuff +# linker +LN= ln +# file deletion command +RM= delete quiet +# library (.lib) file creation command +AR= lb + + +# source files (independently compilable files) +SOURCES= jbsmooth.c jcarith.c jccolor.c jcdeflts.c jcexpand.c jchuff.c \ + jcmain.c jcmaster.c jcmcu.c jcpipe.c jcsample.c jdarith.c jdcolor.c \ + jddeflts.c jdhuff.c jdmain.c jdmaster.c jdmcu.c jdpipe.c jdsample.c \ + jerror.c jquant1.c jquant2.c jfwddct.c jrevdct.c jutils.c \ + jvirtmem.c jrdjfif.c jrdgif.c jrdppm.c jrdrle.c jrdtarga.c \ + jwrjfif.c jwrgif.c jwrppm.c jwrrle.c jwrtarga.c +# files included by source files +INCLUDES= jinclude.h jconfig.h jpegdata.h jversion.h egetopt.c +# documentation, test, and support files +DOCS= README SETUP USAGE CHANGELOG cjpeg.1 djpeg.1 architecture codingrules +MAKEFILES= makefile.ansi makefile.unix makefile.manx makefile.sas \ + makefile.mc5 makefile.mc6 makcjpeg.lnk makdjpeg.lnk makefile.tc \ + makcjpeg.lst makdjpeg.lst makefile.pwc makcjpeg.cf makdjpeg.cf \ + makljpeg.cf +OTHERFILES= ansi2knr.c config.c +TESTFILES= testorig.jpg testimg.ppm testimg.jpg +DISTFILES= $(DOCS) $(MAKEFILES) $(SOURCES) $(INCLUDES) $(OTHERFILES) \ + $(TESTFILES) +# objectfiles common to cjpeg and djpeg +COMOBJECTS= jutils.o jvirtmem.o jerror.o +# compression objectfiles +CLIBOBJECTS= jcmaster.o jcdeflts.o jcarith.o jccolor.o jcexpand.o jchuff.o \ + jcmcu.o jcpipe.o jcsample.o jfwddct.o jwrjfif.o jrdgif.o jrdppm.o \ + jrdrle.o jrdtarga.o +COBJECTS= jcmain.o $(CLIBOBJECTS) $(COMOBJECTS) +# decompression objectfiles +DLIBOBJECTS= jdmaster.o jddeflts.o jbsmooth.o jdarith.o jdcolor.o jdhuff.o \ + jdmcu.o jdpipe.o jdsample.o jquant1.o jquant2.o jrevdct.o jrdjfif.o \ + jwrgif.o jwrppm.o jwrrle.o jwrtarga.o +DOBJECTS= jdmain.o $(DLIBOBJECTS) $(COMOBJECTS) +# These objectfiles are included in libjpeg.lib +LIBOBJECTS= $(CLIBOBJECTS) $(DLIBOBJECTS) $(COMOBJECTS) + + +all: cjpeg djpeg +# By default, libjpeg.lib is not built unless you explicitly request it. +# You can add libjpeg.lib to the line above if you want it built by default. + + +cjpeg: $(COBJECTS) + $(LN) $(LDFLAGS) -o cjpeg $(COBJECTS) $(LDLIBS) + +djpeg: $(DOBJECTS) + $(LN) $(LDFLAGS) -o djpeg $(DOBJECTS) $(LDLIBS) + +# libjpeg.lib is useful if you are including the JPEG software in a larger +# program; you'd include it in your link, rather than the individual modules. +libjpeg.lib: $(LIBOBJECTS) + -$(RM) libjpeg.lib + $(AR) libjpeg.lib $(LIBOBJECTS) + +clean: + -$(RM) *.o cjpeg djpeg libjpeg.lib core testout.ppm testout.jpg + +distribute: + -$(RM) jpegsrc.tar* + tar cvf jpegsrc.tar $(DISTFILES) + compress -v jpegsrc.tar + +test: cjpeg djpeg + -$(RM) testout.ppm testout.jpg + djpeg testorig.jpg testout.ppm + cjpeg testimg.ppm testout.jpg + cmp testimg.ppm testout.ppm + cmp testimg.jpg testout.jpg + + +jbsmooth.o : jbsmooth.c jinclude.h jconfig.h jpegdata.h +jcarith.o : jcarith.c jinclude.h jconfig.h jpegdata.h +jccolor.o : jccolor.c jinclude.h jconfig.h jpegdata.h +jcdeflts.o : jcdeflts.c jinclude.h jconfig.h jpegdata.h +jcexpand.o : jcexpand.c jinclude.h jconfig.h jpegdata.h +jchuff.o : jchuff.c jinclude.h jconfig.h jpegdata.h +jcmain.o : jcmain.c jinclude.h jconfig.h jpegdata.h jversion.h egetopt.c +jcmaster.o : jcmaster.c jinclude.h jconfig.h jpegdata.h +jcmcu.o : jcmcu.c jinclude.h jconfig.h jpegdata.h +jcpipe.o : jcpipe.c jinclude.h jconfig.h jpegdata.h +jcsample.o : jcsample.c jinclude.h jconfig.h jpegdata.h +jdarith.o : jdarith.c jinclude.h jconfig.h jpegdata.h +jdcolor.o : jdcolor.c jinclude.h jconfig.h jpegdata.h +jddeflts.o : jddeflts.c jinclude.h jconfig.h jpegdata.h +jdhuff.o : jdhuff.c jinclude.h jconfig.h jpegdata.h +jdmain.o : jdmain.c jinclude.h jconfig.h jpegdata.h jversion.h egetopt.c +jdmaster.o : jdmaster.c jinclude.h jconfig.h jpegdata.h +jdmcu.o : jdmcu.c jinclude.h jconfig.h jpegdata.h +jdpipe.o : jdpipe.c jinclude.h jconfig.h jpegdata.h +jdsample.o : jdsample.c jinclude.h jconfig.h jpegdata.h +jerror.o : jerror.c jinclude.h jconfig.h jpegdata.h +jquant1.o : jquant1.c jinclude.h jconfig.h jpegdata.h +jquant2.o : jquant2.c jinclude.h jconfig.h jpegdata.h +jfwddct.o : jfwddct.c jinclude.h jconfig.h jpegdata.h +jrevdct.o : jrevdct.c jinclude.h jconfig.h jpegdata.h +jutils.o : jutils.c jinclude.h jconfig.h jpegdata.h +jvirtmem.o : jvirtmem.c jinclude.h jconfig.h jpegdata.h +jrdjfif.o : jrdjfif.c jinclude.h jconfig.h jpegdata.h +jrdgif.o : jrdgif.c jinclude.h jconfig.h jpegdata.h +jrdppm.o : jrdppm.c jinclude.h jconfig.h jpegdata.h +jrdrle.o : jrdrle.c jinclude.h jconfig.h jpegdata.h +jrdtarga.o : jrdtarga.c jinclude.h jconfig.h jpegdata.h +jwrjfif.o : jwrjfif.c jinclude.h jconfig.h jpegdata.h +jwrgif.o : jwrgif.c jinclude.h jconfig.h jpegdata.h +jwrppm.o : jwrppm.c jinclude.h jconfig.h jpegdata.h +jwrrle.o : jwrrle.c jinclude.h jconfig.h jpegdata.h +jwrtarga.o : jwrtarga.c jinclude.h jconfig.h jpegdata.h diff --git a/makefile.mc5 b/makefile.mc5 index eee9f217b..31a6a9b42 100644 --- a/makefile.mc5 +++ b/makefile.mc5 @@ -2,7 +2,7 @@ # This makefile is for Microsoft C for MS-DOS, version 5.x. -# See README and edit jconfig.h before saying "make" !! +# Read SETUP instructions before saying "make" !! # Microsoft's brain-damaged version of make uses nonstandard syntax (a blank # line is needed to terminate a command list) and it simply scans the rules @@ -13,23 +13,48 @@ # from the library. The objectfiles are also kept separately as timestamps. # You may need to adjust these cc options: -CFLAGS= /AS /I. /W3 /Oail /Gs /DMEM_STATS # NB: /Gs turns off stack oflo checks -LDFLAGS= /Fm /F 2000 # /F hhhh sets stack size (in hex) +CFLAGS= /AS /I. /W3 /Oail /Gs # NB: /Gs turns off stack oflo checks +LDFLAGS= /Fm /F 2000 # /F hhhh sets stack size (in hex) # In particular: # Add /DMSDOS if your compiler doesn't automatically #define MSDOS. -# Add /DHAVE_GETOPT if your library includes getopt(3) (see jcmain.c, jdmain.c). -# /DMEM_STATS is optional -- it enables gathering of memory usage statistics. - +# Add /DMEM_STATS to enable gathering of memory usage statistics. +# You might also want to add /G2 if you have an 80286, etc. + + +# source files (independently compilable files) +SOURCES= jbsmooth.c jcarith.c jccolor.c jcdeflts.c jcexpand.c jchuff.c \ + jcmain.c jcmaster.c jcmcu.c jcpipe.c jcsample.c jdarith.c jdcolor.c \ + jddeflts.c jdhuff.c jdmain.c jdmaster.c jdmcu.c jdpipe.c jdsample.c \ + jerror.c jquant1.c jquant2.c jfwddct.c jrevdct.c jutils.c \ + jvirtmem.c jrdjfif.c jrdgif.c jrdppm.c jrdrle.c jrdtarga.c \ + jwrjfif.c jwrgif.c jwrppm.c jwrrle.c jwrtarga.c +# files included by source files +INCLUDES= jinclude.h jconfig.h jpegdata.h jversion.h egetopt.c +# documentation, test, and support files +DOCS= README SETUP USAGE CHANGELOG cjpeg.1 djpeg.1 architecture codingrules +MAKEFILES= makefile.ansi makefile.unix makefile.manx makefile.sas \ + makefile.mc5 makefile.mc6 makcjpeg.lnk makdjpeg.lnk makefile.tc \ + makcjpeg.lst makdjpeg.lst makefile.pwc makcjpeg.cf makdjpeg.cf \ + makljpeg.cf +OTHERFILES= ansi2knr.c config.c +TESTFILES= testorig.jpg testimg.ppm testimg.jpg +DISTFILES= $(DOCS) $(MAKEFILES) $(SOURCES) $(INCLUDES) $(OTHERFILES) \ + $(TESTFILES) +# objectfiles common to cjpeg and djpeg +COMOBJECTS= jutils.obj jvirtmem.obj jerror.obj # compression objectfiles -COBJECTS = jcmain.obj jcmaster.obj jcdeflts.obj jcarith.obj jccolor.obj jcexpand.obj \ - jchuff.obj jcmcu.obj jcpipe.obj jcsample.obj jfwddct.obj \ - jrdgif.obj jrdppm.obj jwrjfif.obj \ - jutils.obj jvirtmem.obj jerror.obj +CLIBOBJECTS= jcmaster.obj jcdeflts.obj jcarith.obj jccolor.obj jcexpand.obj \ + jchuff.obj jcmcu.obj jcpipe.obj jcsample.obj jfwddct.obj \ + jwrjfif.obj jrdgif.obj jrdppm.obj jrdrle.obj jrdtarga.obj +COBJECTS= jcmain.obj $(CLIBOBJECTS) $(COMOBJECTS) # decompression objectfiles -DOBJECTS = jdmain.obj jdmaster.obj jbsmooth.obj jdarith.obj jdcolor.obj jdhuff.obj \ - jdmcu.obj jdpipe.obj jdsample.obj jquant1.obj jquant2.obj jrevdct.obj \ - jrdjfif.obj jwrgif.obj jwrppm.obj \ - jutils.obj jvirtmem.obj jerror.obj +DLIBOBJECTS= jdmaster.obj jddeflts.obj jbsmooth.obj jdarith.obj jdcolor.obj \ + jdhuff.obj jdmcu.obj jdpipe.obj jdsample.obj jquant1.obj \ + jquant2.obj jrevdct.obj jrdjfif.obj jwrgif.obj jwrppm.obj \ + jwrrle.obj jwrtarga.obj +DOBJECTS= jdmain.obj $(DLIBOBJECTS) $(COMOBJECTS) +# These objectfiles are included in libjpeg.lib +LIBOBJECTS= $(CLIBOBJECTS) $(DLIBOBJECTS) $(COMOBJECTS) # inference rule used for all compilations except jcmain.c, jdmain.c @@ -38,74 +63,83 @@ DOBJECTS = jdmain.obj jdmaster.obj jbsmooth.obj jdarith.obj jdcolor.obj jdhuff.o cl $(CFLAGS) /c $*.c lib libjpeg -+$*.obj; -# these two objectfiles are not inserted into libjpeg -# because they have duplicate global symbol names (notably main()). -jcmain.obj: jcmain.c jinclude.h jconfig.h jpegdata.h egetopt.c - cl $(CFLAGS) /c $*.c -jdmain.obj: jdmain.c jinclude.h jconfig.h jpegdata.h egetopt.c +jbsmooth.obj : jbsmooth.c jinclude.h jconfig.h jpegdata.h + +jcarith.obj : jcarith.c jinclude.h jconfig.h jpegdata.h + +jccolor.obj : jccolor.c jinclude.h jconfig.h jpegdata.h + +jcdeflts.obj : jcdeflts.c jinclude.h jconfig.h jpegdata.h + +jcexpand.obj : jcexpand.c jinclude.h jconfig.h jpegdata.h + +jchuff.obj : jchuff.c jinclude.h jconfig.h jpegdata.h + +jcmain.obj : jcmain.c jinclude.h jconfig.h jpegdata.h jversion.h egetopt.c cl $(CFLAGS) /c $*.c +jcmaster.obj : jcmaster.c jinclude.h jconfig.h jpegdata.h -jbsmooth.obj: jbsmooth.c jinclude.h jconfig.h jpegdata.h +jcmcu.obj : jcmcu.c jinclude.h jconfig.h jpegdata.h -jcarith.obj: jcarith.c jinclude.h jconfig.h jpegdata.h +jcpipe.obj : jcpipe.c jinclude.h jconfig.h jpegdata.h -jccolor.obj: jccolor.c jinclude.h jconfig.h jpegdata.h +jcsample.obj : jcsample.c jinclude.h jconfig.h jpegdata.h -jcdeflts.obj: jcdeflts.c jinclude.h jconfig.h jpegdata.h +jdarith.obj : jdarith.c jinclude.h jconfig.h jpegdata.h -jcexpand.obj: jcexpand.c jinclude.h jconfig.h jpegdata.h +jdcolor.obj : jdcolor.c jinclude.h jconfig.h jpegdata.h -jchuff.obj: jchuff.c jinclude.h jconfig.h jpegdata.h +jddeflts.obj : jddeflts.c jinclude.h jconfig.h jpegdata.h -jcmaster.obj: jcmaster.c jinclude.h jconfig.h jpegdata.h +jdhuff.obj : jdhuff.c jinclude.h jconfig.h jpegdata.h -jcmcu.obj: jcmcu.c jinclude.h jconfig.h jpegdata.h +jdmain.obj : jdmain.c jinclude.h jconfig.h jpegdata.h jversion.h egetopt.c + cl $(CFLAGS) /c $*.c -jcpipe.obj: jcpipe.c jinclude.h jconfig.h jpegdata.h +jdmaster.obj : jdmaster.c jinclude.h jconfig.h jpegdata.h -jcsample.obj: jcsample.c jinclude.h jconfig.h jpegdata.h +jdmcu.obj : jdmcu.c jinclude.h jconfig.h jpegdata.h -jdarith.obj: jdarith.c jinclude.h jconfig.h jpegdata.h +jdpipe.obj : jdpipe.c jinclude.h jconfig.h jpegdata.h -jdcolor.obj: jdcolor.c jinclude.h jconfig.h jpegdata.h +jdsample.obj : jdsample.c jinclude.h jconfig.h jpegdata.h -jdhuff.obj: jdhuff.c jinclude.h jconfig.h jpegdata.h +jerror.obj : jerror.c jinclude.h jconfig.h jpegdata.h -jdmaster.obj: jdmaster.c jinclude.h jconfig.h jpegdata.h +jquant1.obj : jquant1.c jinclude.h jconfig.h jpegdata.h -jdmcu.obj: jdmcu.c jinclude.h jconfig.h jpegdata.h +jquant2.obj : jquant2.c jinclude.h jconfig.h jpegdata.h -jdpipe.obj: jdpipe.c jinclude.h jconfig.h jpegdata.h +jfwddct.obj : jfwddct.c jinclude.h jconfig.h jpegdata.h -jdsample.obj: jdsample.c jinclude.h jconfig.h jpegdata.h +jrevdct.obj : jrevdct.c jinclude.h jconfig.h jpegdata.h -jerror.obj: jerror.c jinclude.h jconfig.h jpegdata.h +jutils.obj : jutils.c jinclude.h jconfig.h jpegdata.h -jfwddct.obj: jfwddct.c jinclude.h jconfig.h jpegdata.h +jvirtmem.obj : jvirtmem.c jinclude.h jconfig.h jpegdata.h -jquant1.obj: jquant1.c jinclude.h jconfig.h jpegdata.h +jrdjfif.obj : jrdjfif.c jinclude.h jconfig.h jpegdata.h -jquant2.obj: jquant2.c jinclude.h jconfig.h jpegdata.h +jrdgif.obj : jrdgif.c jinclude.h jconfig.h jpegdata.h -jrdjfif.obj: jrdjfif.c jinclude.h jconfig.h jpegdata.h +jrdppm.obj : jrdppm.c jinclude.h jconfig.h jpegdata.h -jrdgif.obj: jrdgif.c jinclude.h jconfig.h jpegdata.h +jrdrle.obj : jrdrle.c jinclude.h jconfig.h jpegdata.h -jrdppm.obj: jrdppm.c jinclude.h jconfig.h jpegdata.h +jrdtarga.obj : jrdtarga.c jinclude.h jconfig.h jpegdata.h -jrevdct.obj: jrevdct.c jinclude.h jconfig.h jpegdata.h +jwrjfif.obj : jwrjfif.c jinclude.h jconfig.h jpegdata.h -jutils.obj: jutils.c jinclude.h jconfig.h jpegdata.h +jwrgif.obj : jwrgif.c jinclude.h jconfig.h jpegdata.h -jvirtmem.obj: jvirtmem.c jinclude.h jconfig.h jpegdata.h +jwrppm.obj : jwrppm.c jinclude.h jconfig.h jpegdata.h -jwrjfif.obj: jwrjfif.c jinclude.h jconfig.h jpegdata.h +jwrrle.obj : jwrrle.c jinclude.h jconfig.h jpegdata.h -jwrgif.obj: jwrgif.c jinclude.h jconfig.h jpegdata.h +jwrtarga.obj : jwrtarga.c jinclude.h jconfig.h jpegdata.h -jwrppm.obj: jwrppm.c jinclude.h jconfig.h jpegdata.h cjpeg.exe: $(COBJECTS) diff --git a/makefile.mc6 b/makefile.mc6 index b9ca60f4d..8007209c7 100644 --- a/makefile.mc6 +++ b/makefile.mc6 @@ -3,71 +3,100 @@ # This makefile is for Microsoft C for MS-DOS, version 6.x (use NMAKE). # Thanks to Alan Wright and Chris Turner of Olivetti Research Ltd. -# See README and edit jconfig.h before saying "make" !! - -all: cjpeg.exe djpeg.exe +# Read SETUP instructions before saying "make" !! # compiler flags. -D gives a #define to the sources: # -O default optimisation # -W3 warning level 3 # -Za ANSI conformance, defines__STDC__ but undefines far -# and near! +# and near, so we DON'T use it. # -D__STDC__ pretend we have full ANSI compliance. MSC is near # enough anyway # -DMSDOS we are on an MSDOS machine # -DMEM_STATS enable memory usage statistics (optional) -# -DHAVE_GETOPT library has getopt routine to parse cmnd line options # -c compile, don't link (implicit in inference rules) +# You might also want to add -G2 if you have an 80286, etc. -CFLAGS = -c -O -W3 -DMSDOS -D__STDC__ -DMEM_STATS +CFLAGS = -c -O -W3 -D__STDC__ -DMSDOS +# source files (independently compilable files) +SOURCES= jbsmooth.c jcarith.c jccolor.c jcdeflts.c jcexpand.c jchuff.c \ + jcmain.c jcmaster.c jcmcu.c jcpipe.c jcsample.c jdarith.c jdcolor.c \ + jddeflts.c jdhuff.c jdmain.c jdmaster.c jdmcu.c jdpipe.c jdsample.c \ + jerror.c jquant1.c jquant2.c jfwddct.c jrevdct.c jutils.c \ + jvirtmem.c jrdjfif.c jrdgif.c jrdppm.c jrdrle.c jrdtarga.c \ + jwrjfif.c jwrgif.c jwrppm.c jwrrle.c jwrtarga.c +# files included by source files +INCLUDES= jinclude.h jconfig.h jpegdata.h jversion.h egetopt.c +# documentation, test, and support files +DOCS= README SETUP USAGE CHANGELOG cjpeg.1 djpeg.1 architecture codingrules +MAKEFILES= makefile.ansi makefile.unix makefile.manx makefile.sas \ + makefile.mc5 makefile.mc6 makcjpeg.lnk makdjpeg.lnk makefile.tc \ + makcjpeg.lst makdjpeg.lst makefile.pwc makcjpeg.cf makdjpeg.cf \ + makljpeg.cf +OTHERFILES= ansi2knr.c config.c +TESTFILES= testorig.jpg testimg.ppm testimg.jpg +DISTFILES= $(DOCS) $(MAKEFILES) $(SOURCES) $(INCLUDES) $(OTHERFILES) \ + $(TESTFILES) +# objectfiles common to cjpeg and djpeg +COMOBJECTS= jutils.obj jvirtmem.obj jerror.obj # compression objectfiles -COBJECTS = jcmain.obj jcmaster.obj jcdeflts.obj jcarith.obj jccolor.obj jcexpand.obj \ - jchuff.obj jcmcu.obj jcpipe.obj jcsample.obj jfwddct.obj \ - jrdgif.obj jrdppm.obj jwrjfif.obj \ - jutils.obj jvirtmem.obj jerror.obj +CLIBOBJECTS= jcmaster.obj jcdeflts.obj jcarith.obj jccolor.obj jcexpand.obj \ + jchuff.obj jcmcu.obj jcpipe.obj jcsample.obj jfwddct.obj \ + jwrjfif.obj jrdgif.obj jrdppm.obj jrdrle.obj jrdtarga.obj +COBJECTS= jcmain.obj $(CLIBOBJECTS) $(COMOBJECTS) # decompression objectfiles -DOBJECTS = jdmain.obj jdmaster.obj jbsmooth.obj jdarith.obj jdcolor.obj jdhuff.obj \ - jdmcu.obj jdpipe.obj jdsample.obj jquant1.obj jquant2.obj jrevdct.obj \ - jrdjfif.obj jwrgif.obj jwrppm.obj \ - jutils.obj jvirtmem.obj jerror.obj +DLIBOBJECTS= jdmaster.obj jddeflts.obj jbsmooth.obj jdarith.obj jdcolor.obj \ + jdhuff.obj jdmcu.obj jdpipe.obj jdsample.obj jquant1.obj \ + jquant2.obj jrevdct.obj jrdjfif.obj jwrgif.obj jwrppm.obj \ + jwrrle.obj jwrtarga.obj +DOBJECTS= jdmain.obj $(DLIBOBJECTS) $(COMOBJECTS) +# These objectfiles are included in libjpeg.lib +LIBOBJECTS= $(CLIBOBJECTS) $(DLIBOBJECTS) $(COMOBJECTS) +all: cjpeg.exe djpeg.exe + # default rules in nmake will use cflags and compile the list below -jbsmooth.o : jbsmooth.c jinclude.h jconfig.h jpegdata.h -jcarith.o : jcarith.c jinclude.h jconfig.h jpegdata.h -jccolor.o : jccolor.c jinclude.h jconfig.h jpegdata.h -jcdeflts.o : jcdeflts.c jinclude.h jconfig.h jpegdata.h -jcexpand.o : jcexpand.c jinclude.h jconfig.h jpegdata.h -jchuff.o : jchuff.c jinclude.h jconfig.h jpegdata.h -jcmain.o : jcmain.c jinclude.h jconfig.h jpegdata.h egetopt.c -jcmaster.o : jcmaster.c jinclude.h jconfig.h jpegdata.h -jcmcu.o : jcmcu.c jinclude.h jconfig.h jpegdata.h -jcpipe.o : jcpipe.c jinclude.h jconfig.h jpegdata.h -jcsample.o : jcsample.c jinclude.h jconfig.h jpegdata.h -jdarith.o : jdarith.c jinclude.h jconfig.h jpegdata.h -jdcolor.o : jdcolor.c jinclude.h jconfig.h jpegdata.h -jdhuff.o : jdhuff.c jinclude.h jconfig.h jpegdata.h -jdmain.o : jdmain.c jinclude.h jconfig.h jpegdata.h egetopt.c -jdmaster.o : jdmaster.c jinclude.h jconfig.h jpegdata.h -jdmcu.o : jdmcu.c jinclude.h jconfig.h jpegdata.h -jdpipe.o : jdpipe.c jinclude.h jconfig.h jpegdata.h -jdsample.o : jdsample.c jinclude.h jconfig.h jpegdata.h -jerror.o : jerror.c jinclude.h jconfig.h jpegdata.h -jfwddct.o : jfwddct.c jinclude.h jconfig.h jpegdata.h -jquant1.o : jquant1.c jinclude.h jconfig.h jpegdata.h -jquant2.o : jquant2.c jinclude.h jconfig.h jpegdata.h -jrdjfif.o : jrdjfif.c jinclude.h jconfig.h jpegdata.h -jrdgif.o : jrdgif.c jinclude.h jconfig.h jpegdata.h -jrdppm.o : jrdppm.c jinclude.h jconfig.h jpegdata.h -jrevdct.o : jrevdct.c jinclude.h jconfig.h jpegdata.h -jutils.o : jutils.c jinclude.h jconfig.h jpegdata.h -jvirtmem.o : jvirtmem.c jinclude.h jconfig.h jpegdata.h -jwrjfif.o : jwrjfif.c jinclude.h jconfig.h jpegdata.h -jwrgif.o : jwrgif.c jinclude.h jconfig.h jpegdata.h -jwrppm.o : jwrppm.c jinclude.h jconfig.h jpegdata.h +jbsmooth.obj : jbsmooth.c jinclude.h jconfig.h jpegdata.h +jcarith.obj : jcarith.c jinclude.h jconfig.h jpegdata.h +jccolor.obj : jccolor.c jinclude.h jconfig.h jpegdata.h +jcdeflts.obj : jcdeflts.c jinclude.h jconfig.h jpegdata.h +jcexpand.obj : jcexpand.c jinclude.h jconfig.h jpegdata.h +jchuff.obj : jchuff.c jinclude.h jconfig.h jpegdata.h +jcmain.obj : jcmain.c jinclude.h jconfig.h jpegdata.h jversion.h egetopt.c +jcmaster.obj : jcmaster.c jinclude.h jconfig.h jpegdata.h +jcmcu.obj : jcmcu.c jinclude.h jconfig.h jpegdata.h +jcpipe.obj : jcpipe.c jinclude.h jconfig.h jpegdata.h +jcsample.obj : jcsample.c jinclude.h jconfig.h jpegdata.h +jdarith.obj : jdarith.c jinclude.h jconfig.h jpegdata.h +jdcolor.obj : jdcolor.c jinclude.h jconfig.h jpegdata.h +jddeflts.obj : jddeflts.c jinclude.h jconfig.h jpegdata.h +jdhuff.obj : jdhuff.c jinclude.h jconfig.h jpegdata.h +jdmain.obj : jdmain.c jinclude.h jconfig.h jpegdata.h jversion.h egetopt.c +jdmaster.obj : jdmaster.c jinclude.h jconfig.h jpegdata.h +jdmcu.obj : jdmcu.c jinclude.h jconfig.h jpegdata.h +jdpipe.obj : jdpipe.c jinclude.h jconfig.h jpegdata.h +jdsample.obj : jdsample.c jinclude.h jconfig.h jpegdata.h +jerror.obj : jerror.c jinclude.h jconfig.h jpegdata.h +jquant1.obj : jquant1.c jinclude.h jconfig.h jpegdata.h +jquant2.obj : jquant2.c jinclude.h jconfig.h jpegdata.h +jfwddct.obj : jfwddct.c jinclude.h jconfig.h jpegdata.h +jrevdct.obj : jrevdct.c jinclude.h jconfig.h jpegdata.h +jutils.obj : jutils.c jinclude.h jconfig.h jpegdata.h +jvirtmem.obj : jvirtmem.c jinclude.h jconfig.h jpegdata.h +jrdjfif.obj : jrdjfif.c jinclude.h jconfig.h jpegdata.h +jrdgif.obj : jrdgif.c jinclude.h jconfig.h jpegdata.h +jrdppm.obj : jrdppm.c jinclude.h jconfig.h jpegdata.h +jrdrle.obj : jrdrle.c jinclude.h jconfig.h jpegdata.h +jrdtarga.obj : jrdtarga.c jinclude.h jconfig.h jpegdata.h +jwrjfif.obj : jwrjfif.c jinclude.h jconfig.h jpegdata.h +jwrgif.obj : jwrgif.c jinclude.h jconfig.h jpegdata.h +jwrppm.obj : jwrppm.c jinclude.h jconfig.h jpegdata.h +jwrrle.obj : jwrrle.c jinclude.h jconfig.h jpegdata.h +jwrtarga.obj : jwrtarga.c jinclude.h jconfig.h jpegdata.h # use linker response files because file list > 128 chars diff --git a/makefile.pwc b/makefile.pwc index f41dd480d..f89e84099 100644 --- a/makefile.pwc +++ b/makefile.pwc @@ -4,22 +4,19 @@ # and Dan Grayson's pd make 2.14 under MS-DOS. # Thanks to Bob Hardy for this version. -# See README and edit jconfig.h before saying "make" !! - -# NOTE: make sure you have converted end-of-line markers to CR/LF in this file -# and in the three mak*.cf files; otherwise pd make and the Mix linker will -# choke. Power C doesn't seem to care whether end-of-lines are CR/LF or just -# LF in the *.h and *.c files. If you blindly converted LF to CR/LF in ALL -# the files, then you broke the test*.* files, which contain binary data. +# Read SETUP instructions before saying "make" !! +# The name of your C compiler: CC=pc # You may need to adjust these cc options: MODEL=m -CFLAGS=-dMEM_STATS -dMSDOS -m$(MODEL) -LDFLAGS= +CFLAGS= -dMSDOS -m$(MODEL) # In particular: -# -dMEM_STATS is optional -- it enables gathering of memory usage statistics. +# Add -dMEM_STATS to enable gathering of memory usage statistics. + +# Link-time cc options: +LDFLAGS= LDLIBS= # miscellaneous OS-dependent stuff @@ -31,16 +28,31 @@ RM=del AR=merge +# source files (independently compilable files) +SOURCES= jbsmooth.c jcarith.c jccolor.c jcdeflts.c jcexpand.c jchuff.c jcmain.c jcmaster.c jcmcu.c jcpipe.c jcsample.c jdarith.c jdcolor.c jddeflts.c jdhuff.c jdmain.c jdmaster.c jdmcu.c jdpipe.c jdsample.c jerror.c jquant1.c jquant2.c jfwddct.c jrevdct.c jutils.c jvirtmem.c jrdjfif.c jrdgif.c jrdppm.c jrdrle.c jrdtarga.c jwrjfif.c jwrgif.c jwrppm.c jwrrle.c jwrtarga.c +# files included by source files +INCLUDES= jinclude.h jconfig.h jpegdata.h jversion.h egetopt.c +# documentation, test, and support files +DOCS= README SETUP USAGE CHANGELOG cjpeg.1 djpeg.1 architecture codingrules +MAKEFILES= makefile.ansi makefile.unix makefile.manx makefile.sas makefile.mc5 makefile.mc6 makcjpeg.lnk makdjpeg.lnk makefile.tc makcjpeg.lst makdjpeg.lst makefile.pwc makcjpeg.cf makdjpeg.cf makljpeg.cf +OTHERFILES= ansi2knr.c config.c +TESTFILES= testorig.jpg testimg.ppm testimg.jpg +DISTFILES= $(DOCS) $(MAKEFILES) $(SOURCES) $(INCLUDES) $(OTHERFILES) $(TESTFILES) +# objectfiles common to cjpeg and djpeg +COMOBJECTS= jutils.mix jvirtmem.mix jerror.mix # compression objectfiles -COBJECTS = jcmain.mix jcmaster.mix jcdeflts.mix jcarith.mix jccolor.mix jcexpand.mix jchuff.mix jcmcu.mix jcpipe.mix jcsample.mix jfwddct.mix jrdgif.mix jrdppm.mix jwrjfif.mix jutils.mix jvirtmem.mix jerror.mix +CLIBOBJECTS= jcmaster.mix jcdeflts.mix jcarith.mix jccolor.mix jcexpand.mix jchuff.mix jcmcu.mix jcpipe.mix jcsample.mix jfwddct.mix jwrjfif.mix jrdgif.mix jrdppm.mix jrdrle.mix jrdtarga.mix +COBJECTS= jcmain.mix $(CLIBOBJECTS) $(COMOBJECTS) # decompression objectfiles -DOBJECTS = jdmain.mix jdmaster.mix jbsmooth.mix jdarith.mix jdcolor.mix jdhuff.mix jdmcu.mix jdpipe.mix jdsample.mix jquant1.mix jquant2.mix jrevdct.mix jrdjfif.mix jwrgif.mix jwrppm.mix jutils.mix jvirtmem.mix jerror.mix -# These objectfiles are included in libjpeg.mix (all but jcmain, jdmain) -LIBOBJECTS = jcmaster.mix jcdeflts.mix jcarith.mix jccolor.mix jcexpand.mix jchuff.mix jcmcu.mix jcpipe.mix jcsample.mix jfwddct.mix jrdgif.mix jrdppm.mix jwrjfif.mix jdmaster.mix jbsmooth.mix jdarith.mix jdcolor.mix jdhuff.mix jdmcu.mix jdpipe.mix jdsample.mix jquant1.mix jquant2.mix jrevdct.mix jrdjfif.mix jwrgif.mix jwrppm.mix jutils.mix jvirtmem.mix jerror.mix +DLIBOBJECTS= jdmaster.mix jddeflts.mix jbsmooth.mix jdarith.mix jdcolor.mix jdhuff.mix jdmcu.mix jdpipe.mix jdsample.mix jquant1.mix jquant2.mix jrevdct.mix jrdjfif.mix jwrgif.mix jwrppm.mix jwrrle.mix jwrtarga.mix +DOBJECTS= jdmain.mix $(DLIBOBJECTS) $(COMOBJECTS) +# These objectfiles are included in libjpeg.mix +LIBOBJECTS= $(CLIBOBJECTS) $(DLIBOBJECTS) $(COMOBJECTS) -all: cjpeg.exe djpeg.exe test +all: cjpeg.exe djpeg.exe # By default, libjpeg.mix is not built unless you explicitly request it. +# You can add libjpeg.mix to the line above if you want it built by default. cjpeg.exe: $(COBJECTS) @@ -66,35 +78,40 @@ test: fc testimg.jpg testout.jpg -jbsmooth.mix : jbsmooth.c jinclude.h jconfig.h jpegdata.h -jcarith.mix : jcarith.c jinclude.h jconfig.h jpegdata.h -jccolor.mix : jccolor.c jinclude.h jconfig.h jpegdata.h -jcdeflts.mix : jcdeflts.c jinclude.h jconfig.h jpegdata.h -jcexpand.mix : jcexpand.c jinclude.h jconfig.h jpegdata.h -jchuff.mix : jchuff.c jinclude.h jconfig.h jpegdata.h -jcmain.mix : jcmain.c jinclude.h jconfig.h jpegdata.h egetopt.c -jcmaster.mix : jcmaster.c jinclude.h jconfig.h jpegdata.h -jcmcu.mix : jcmcu.c jinclude.h jconfig.h jpegdata.h -jcpipe.mix : jcpipe.c jinclude.h jconfig.h jpegdata.h -jcsample.mix : jcsample.c jinclude.h jconfig.h jpegdata.h -jdarith.mix : jdarith.c jinclude.h jconfig.h jpegdata.h -jdcolor.mix : jdcolor.c jinclude.h jconfig.h jpegdata.h -jdhuff.mix : jdhuff.c jinclude.h jconfig.h jpegdata.h -jdmain.mix : jdmain.c jinclude.h jconfig.h jpegdata.h egetopt.c -jdmaster.mix : jdmaster.c jinclude.h jconfig.h jpegdata.h -jdmcu.mix : jdmcu.c jinclude.h jconfig.h jpegdata.h -jdpipe.mix : jdpipe.c jinclude.h jconfig.h jpegdata.h -jdsample.mix : jdsample.c jinclude.h jconfig.h jpegdata.h -jerror.mix : jerror.c jinclude.h jconfig.h jpegdata.h -jfwddct.mix : jfwddct.c jinclude.h jconfig.h jpegdata.h -jquant1.mix : jquant1.c jinclude.h jconfig.h jpegdata.h -jquant2.mix : jquant2.c jinclude.h jconfig.h jpegdata.h -jrdjfif.mix : jrdjfif.c jinclude.h jconfig.h jpegdata.h -jrdgif.mix : jrdgif.c jinclude.h jconfig.h jpegdata.h -jrdppm.mix : jrdppm.c jinclude.h jconfig.h jpegdata.h -jrevdct.mix : jrevdct.c jinclude.h jconfig.h jpegdata.h -jutils.mix : jutils.c jinclude.h jconfig.h jpegdata.h -jvirtmem.mix : jvirtmem.c jinclude.h jconfig.h jpegdata.h -jwrjfif.mix : jwrjfif.c jinclude.h jconfig.h jpegdata.h -jwrgif.mix : jwrgif.c jinclude.h jconfig.h jpegdata.h -jwrppm.mix : jwrppm.c jinclude.h jconfig.h jpegdata.h +jbsmooth.mix : jbsmooth.c jinclude.h jconfig.h jpegdata.h +jcarith.mix : jcarith.c jinclude.h jconfig.h jpegdata.h +jccolor.mix : jccolor.c jinclude.h jconfig.h jpegdata.h +jcdeflts.mix : jcdeflts.c jinclude.h jconfig.h jpegdata.h +jcexpand.mix : jcexpand.c jinclude.h jconfig.h jpegdata.h +jchuff.mix : jchuff.c jinclude.h jconfig.h jpegdata.h +jcmain.mix : jcmain.c jinclude.h jconfig.h jpegdata.h jversion.h egetopt.c +jcmaster.mix : jcmaster.c jinclude.h jconfig.h jpegdata.h +jcmcu.mix : jcmcu.c jinclude.h jconfig.h jpegdata.h +jcpipe.mix : jcpipe.c jinclude.h jconfig.h jpegdata.h +jcsample.mix : jcsample.c jinclude.h jconfig.h jpegdata.h +jdarith.mix : jdarith.c jinclude.h jconfig.h jpegdata.h +jdcolor.mix : jdcolor.c jinclude.h jconfig.h jpegdata.h +jddeflts.mix : jddeflts.c jinclude.h jconfig.h jpegdata.h +jdhuff.mix : jdhuff.c jinclude.h jconfig.h jpegdata.h +jdmain.mix : jdmain.c jinclude.h jconfig.h jpegdata.h jversion.h egetopt.c +jdmaster.mix : jdmaster.c jinclude.h jconfig.h jpegdata.h +jdmcu.mix : jdmcu.c jinclude.h jconfig.h jpegdata.h +jdpipe.mix : jdpipe.c jinclude.h jconfig.h jpegdata.h +jdsample.mix : jdsample.c jinclude.h jconfig.h jpegdata.h +jerror.mix : jerror.c jinclude.h jconfig.h jpegdata.h +jquant1.mix : jquant1.c jinclude.h jconfig.h jpegdata.h +jquant2.mix : jquant2.c jinclude.h jconfig.h jpegdata.h +jfwddct.mix : jfwddct.c jinclude.h jconfig.h jpegdata.h +jrevdct.mix : jrevdct.c jinclude.h jconfig.h jpegdata.h +jutils.mix : jutils.c jinclude.h jconfig.h jpegdata.h +jvirtmem.mix : jvirtmem.c jinclude.h jconfig.h jpegdata.h +jrdjfif.mix : jrdjfif.c jinclude.h jconfig.h jpegdata.h +jrdgif.mix : jrdgif.c jinclude.h jconfig.h jpegdata.h +jrdppm.mix : jrdppm.c jinclude.h jconfig.h jpegdata.h +jrdrle.mix : jrdrle.c jinclude.h jconfig.h jpegdata.h +jrdtarga.mix : jrdtarga.c jinclude.h jconfig.h jpegdata.h +jwrjfif.mix : jwrjfif.c jinclude.h jconfig.h jpegdata.h +jwrgif.mix : jwrgif.c jinclude.h jconfig.h jpegdata.h +jwrppm.mix : jwrppm.c jinclude.h jconfig.h jpegdata.h +jwrrle.mix : jwrrle.c jinclude.h jconfig.h jpegdata.h +jwrtarga.mix : jwrtarga.c jinclude.h jconfig.h jpegdata.h diff --git a/makefile.sas b/makefile.sas new file mode 100644 index 000000000..877edc538 --- /dev/null +++ b/makefile.sas @@ -0,0 +1,152 @@ +# Makefile for Independent JPEG Group's software + +# This makefile is for Amiga systems using SAS C 5.10b. +# Contributed by Ed Hanway (sisd!jeh@uunet.uu.net). + +# Read SETUP instructions before saying "make" !! + +# The name of your C compiler: +CC= lc + +# Uncomment the following lines for generic 680x0 version +ARCHFLAGS= +SUFFIX= + +# Uncomment the following lines for 68030-only version +#ARCHFLAGS= -m3 +#SUFFIX=.030 + +# You may need to adjust these cc options: +CFLAGS= -v -b -rr -O -j104 -D__STDC__ -DTWO_FILE_COMMANDLINE -DINCOMPLETE_TYPES_BROKEN $(ARCHFLAGS) +# -j104 disables warnings for mismatched const qualifiers + +# Link-time cc options: +LDFLAGS= SC SD ND BATCH + +# To link any special libraries, add the necessary commands here. +LDLIBS= LIB LIB:lcr.lib + +# miscellaneous OS-dependent stuff +# linker +LN= blink +# file deletion command +RM= delete quiet +# library (.lib) file creation command +AR= oml + + +# source files (independently compilable files) +SOURCES= jbsmooth.c jcarith.c jccolor.c jcdeflts.c jcexpand.c jchuff.c \ + jcmain.c jcmaster.c jcmcu.c jcpipe.c jcsample.c jdarith.c jdcolor.c \ + jddeflts.c jdhuff.c jdmain.c jdmaster.c jdmcu.c jdpipe.c jdsample.c \ + jerror.c jquant1.c jquant2.c jfwddct.c jrevdct.c jutils.c \ + jvirtmem.c jrdjfif.c jrdgif.c jrdppm.c jrdrle.c jrdtarga.c \ + jwrjfif.c jwrgif.c jwrppm.c jwrrle.c jwrtarga.c +# files included by source files +INCLUDES= jinclude.h jconfig.h jpegdata.h jversion.h egetopt.c +# documentation, test, and support files +DOCS= README SETUP USAGE CHANGELOG cjpeg.1 djpeg.1 architecture codingrules +MAKEFILES= makefile.ansi makefile.unix makefile.manx makefile.sas \ + makefile.mc5 makefile.mc6 makcjpeg.lnk makdjpeg.lnk makefile.tc \ + makcjpeg.lst makdjpeg.lst makefile.pwc makcjpeg.cf makdjpeg.cf \ + makljpeg.cf +OTHERFILES= ansi2knr.c config.c +TESTFILES= testorig.jpg testimg.ppm testimg.jpg +DISTFILES= $(DOCS) $(MAKEFILES) $(SOURCES) $(INCLUDES) $(OTHERFILES) \ + $(TESTFILES) +# objectfiles common to cjpeg and djpeg +COMOBJECTS= jutils.o jvirtmem.o jerror.o +# compression objectfiles +CLIBOBJECTS= jcmaster.o jcdeflts.o jcarith.o jccolor.o jcexpand.o jchuff.o \ + jcmcu.o jcpipe.o jcsample.o jfwddct.o jwrjfif.o jrdgif.o jrdppm.o \ + jrdrle.o jrdtarga.o +COBJECTS= jcmain.o $(CLIBOBJECTS) $(COMOBJECTS) +# decompression objectfiles +DLIBOBJECTS= jdmaster.o jddeflts.o jbsmooth.o jdarith.o jdcolor.o jdhuff.o \ + jdmcu.o jdpipe.o jdsample.o jquant1.o jquant2.o jrevdct.o jrdjfif.o \ + jwrgif.o jwrppm.o jwrrle.o jwrtarga.o +DOBJECTS= jdmain.o $(DLIBOBJECTS) $(COMOBJECTS) +# These objectfiles are included in libjpeg.lib +LIBOBJECTS= $(CLIBOBJECTS) $(DLIBOBJECTS) $(COMOBJECTS) + + +all: cjpeg$(SUFFIX) djpeg$(SUFFIX) +# By default, libjpeg.lib is not built unless you explicitly request it. +# You can add libjpeg.lib to the line above if you want it built by default. + + +cjpeg$(SUFFIX): $(COBJECTS) + $(LN) V<#=b^FvwQj1*yrbizjAZ*%#{Hs|Qh2u} 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 literal 49167 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 diff --git a/testorig.jpg b/testorig.jpg index 1a628fcce7734217234069f0c292be786d1de130..d37b1da6de8aa36921997d3ffa1552ada8d62cb2 100644 GIT binary patch delta 4003 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 delta 2534 zcma*dcQhLa0|4;Ej;d80(X!g~GWPYNMfxKQFF1LM4dNC7|@ zSy@GSMSU<>Kfv0=+B)F>g#O?AJ_wKz10(^G1c9mm0U4m64DdS#SP1|Cg@OM8_%EP< zppdYLs2E86OzwPpel`%2-voS4A0S2k&667)M3Y6A`;wTFj~8@5qZ;?xzQ&@KHc=oG zz~PgdT6$$i!yyc@rboWsv2wQJg;Q=R2=qvo7!x><`|F}4U^lEh{_OC2ksHWla0lPv zlrZ)#+)sL_>i2|anP-EuuQydbbrI8@%?!t?{8;OX;Soy_L6*9jct#?<=NUu1_L1@# zRdaC9c92GJ^l>;tmvBA1cks!dra9Q;!+QO;DCEh!-9g2rpgcHV&~Fset8$CU*c8np zPd~uwyNWYnDuqa)psh$tOifP|d93{8A8OMljaF#~G{3aFA+>Uo)mY36y1*|AV@~R8v&7^QahUU$ zI{27VZE*VV+-U&e+F?B=Q4x=ZZ0|G<`u4*qxmee9>#E!1QVCQ~-?A~i9cZ**64ZBj zcKgxY!=-KpaFGcoNO3EUJ?a3;9-mL9_=)1v5w=}M`-XqR91hhk10YMQ8G^*5=`O1s zcl1O}E15oXq;WBhG}E}ZY&NI12zEfc6b85#!wP!jC8m{PLwmr(OuxySMzES%?}sbr z+G9+I-BFp4aR-W*?pVg<)?ZU3;E>4BdTGWj2+ZfLbY*@=E%4UMp@@BPdH{#)tO(JDOy8+bvD+xuE*A^y+DfsBIPUBnvH0N35hx&2+Fiw$N)@ z7N9rnnQfB=qK#L6#&u5XfL0^#f19TBrm7u48V?+zA(pA4U*=o+BgA|iqjS#ycSOnc zln`}O_Y?w}Tl6kh)_OdqUF4wtxlm-(uv+mXxm)>hEt1_LTY5ZWJ3(FP%~Aih9bmdM zU#9yi#FfitMVLufH)eipd(|s%jYpf!aWv}CC?tCX(I3sXG2q?_BZS+=(xnUGktTkV z*BJ^|2?BN%WXxdg-+_wKqR*|6uHF0)q+3c^D#Imb_ zGDCvLa|zW$!yvc#s)`8i4Xc|I7xp8jEvyaMWb4JeKF&qBgr`qcYdICnLthTkdQWZ8 z=$C}KE6e1;d{T%P(#b6PjML`GM0W8}`(j5SM|V{IC~egdNuGmU3dq?v^aAxmQP+b| zT_i~0&k;j)H(D1l%m$0Er-J zxm*#iBHkTg@FFv}usT)ary?3*LBZ-=a!KqtFSVSdiQ8{8FR9Q4gmsoZTNm@lookX= zpK?={2BHRXqI-Sqz)8gXocpr_aM~0g>R4Q@Aw9p}X#f7tDtR%Lj4o8SpNFm(|CdLR zr(&bDEpl3yUN&drAL4Ij9%EGdIC<`&S=B$xEo0F6IQz}T-lbxtDUtl;Fn^sGUCr4W zEj1Wtitc$}W;0z|ARnxKRSR;~t;Cz%p%xf(NA2!=~6? z-Vc?gjn3EPiY`0uH@q!KibSx(sNZ~({1Jukb!GBLJe`#z^#*58eI_+1_1>A;duC&oGIX22uhxR~In&rC zlgfFxkFEtHg9lLqeLVdx+cQCZ;2~leV&6vZ7H%iDUc$eH{oK<%KewbTi@fm;8{j7G z|9DfAUR{u`P6gKG8zPwSDmEEupJ#{1aoG1dGwx4aO+%-P7{13DWB8j98=#D{V`B$3 zR%=~w%X8nflhx~cYN1f`bx-_f(RNR2J7=1WuF6K!Zp+X;Fb`)?bOIaJ{g8jEyMAJgzKGfvHq8hGcIO#&TXI%^zk+R&W>D*Am$gnC$Qd?sEX9B^$h9rH1syi9uDn3qjND&8 zMR07w44oaa>qeE(E;qbnYgI(1R1;-g0mlICR;HfPE6HKjx^s6x%4-iz`|MW_)Lo#g z3rQ5LMwADn2+1Fb#XB`QH2QKo+}LXgJZ(L*Pu8{=TMBdqBVz#CO~9wl6gnCg(_rPr nMvGkxXAK=jbf;psdfhJ0a0-va5Nir=zKUCuM)73n-~ad*^~{!o