nuttx-apps/graphics/tiff/tiff_initialize.c
Alin Jerpelea 39c4ef6ca2 apps: graphics: tiff: update licenses to Apache
Gregory Nutt is the copyright holder for those files and he has submitted the
SGA as a result we can migrate the licenses to Apache.

Signed-off-by: Alin Jerpelea <alin.jerpelea@sony.com>
2022-03-11 00:14:32 +02:00

898 lines
29 KiB
C

/****************************************************************************
* apps/graphics/tiff/tiff_initialize.c
*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership. The
* ASF licenses this file to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance with the
* License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
* License for the specific language governing permissions and limitations
* under the License.
*
****************************************************************************/
/****************************************************************************
* Included Files
****************************************************************************/
#include <nuttx/config.h>
#include <string.h>
#include <unistd.h>
#include <fcntl.h>
#include <time.h>
#include <assert.h>
#include <errno.h>
#include <debug.h>
#include "graphics/tiff.h"
#include "tiff_internal.h"
/****************************************************************************
* Pre-Processor Definitions
****************************************************************************/
/* Bi-level Images
*
* Offset Description Contents/Notes
* Header: 0 Byte Order "II" or "MM"
* 2 Magic Number 42
* 4 1st IFD offset 10
* 8 [2 bytes padding]
* IFD: 10 Number of Directory Entries 13
* 12 NewSubfileType
* 24 ImageWidth Number of columns is a user parameter
* 36 ImageLength Number of rows is a user parameter
* 48 Compression Hard-coded no compression (for now)
* 60 PhotometricInterpretation Value is a user parameter
* 72 StripOffsets Offset and count determined as strips added
* 84 RowsPerStrip Value is a user parameter
* 96 StripByteCounts Offset and count determined as strips added
* 108 XResolution Value is a user parameter
* 120 YResolution Value is a user parameter
* 132 Resolution Unit Hard-coded to "inches"
* 144 Software
* 156 DateTime
* 168 Next IFD offset 0
* 170 [2 bytes padding]
* Values:
* 172 XResolution Hard-coded to 300/1
* 180 YResolution Hard-coded to 300/1
* 188 "NuttX" Length = 6 (including NUL terminator)
* 194 "YYYY:MM:DD HH:MM:SS" Length = 20 (ncluding NUL terminator)
* 214 [2 bytes padding]
* 216 StripByteCounts Beginning of strip byte counts
* xxx StripOffsets Beginning of strip offsets
* xxx [Probably padding]
* xxx Data for strips Beginning of strip data
*/
#define TIFF_IFD_OFFSET (SIZEOF_TIFF_HEADER+2)
#define TIFF_BILEV_NIFDENTRIES 13
#define TIFF_BILEV_STRIPIFDOFFS 72
#define TIFF_BILEV_STRIPBCIFDOFFS 96
#define TIFF_BILEV_VALOFFSET 172
#define TIFF_BILEV_XRESOFFSET 172
#define TIFF_BILEV_YRESOFFSET 180
#define TIFF_BILEV_SWOFFSET 188
#define TIFF_BILEV_DATEOFFSET 194
#define TIFF_BILEV_STRIPBCOFFSET 216
#define TIFF_SOFTWARE_STRING "NuttX"
#define TIFF_SOFTWARE_STRLEN 6
#define TIFF_DATETIME_FORMAT "%Y:%m:%d %H:%M:%S"
#define TIFF_DATETIME_STRLEN 20
/* Greyscale Images have one additional IFD entry: BitsPerSample (4 or 8)
*
* Header: 0 Byte Order "II" or "MM"
* 2 Magic Number 42
* 4 1st IFD offset 10
* 8 [2 bytes padding]
* IFD: 10 Number of Directory Entries 14
* 12 NewSubfileType
* 24 ImageWidth Number of columns is a user parameter
* 36 ImageLength Number of rows is a user parameter
* 48 BitsPerSample
* 60 Compression Hard-coded no compression (for now)
* 72 PhotometricInterpretation Value is a user parameter
* 84 StripOffsets Offset and count determined as strips added
* 96 RowsPerStrip Value is a user parameter
* 108 StripByteCounts Offset and count determined as strips added
* 120 XResolution Value is a user parameter
* 132 YResolution Value is a user parameter
* 144 Resolution Unit Hard-coded to "inches"
* 156 Software
* 168 DateTime
* 180 Next IFD offset 0
* 182 [2 bytes padding]
* Values:
* 184 XResolution Hard-coded to 300/1
* 192 YResolution Hard-coded to 300/1
* 200 "NuttX" Length = 6 (including NUL terminator)
* 206 "YYYY:MM:DD HH:MM:SS" Length = 20 (ncluding NUL terminator)
* 226 [2 bytes padding]
* 228 StripByteCounts Beginning of strip byte counts
* xxx StripOffsets Beginning of strip offsets
* xxx [Probably padding]
* xxx Data for strips Beginning of strip data
*/
#define TIFF_GREY_NIFDENTRIES 14
#define TIFF_GREY_STRIPIFDOFFS 84
#define TIFF_GREY_STRIPBCIFDOFFS 108
#define TIFF_GREY_VALOFFSET 184
#define TIFF_GREY_XRESOFFSET 184
#define TIFF_GREY_YRESOFFSET 192
#define TIFF_GREY_SWOFFSET 200
#define TIFF_GREY_DATEOFFSET 206
#define TIFF_GREY_STRIPBCOFFSET 228
/* RGB Images have two additional IFD entries: BitsPerSample (8,8,8) and
* SamplesPerPixel (3):
*
* Header: 0 Byte Order "II" or "MM"
* 2 Magic Number 42
* 4 1st IFD offset 10
* 8 [2 bytes padding]
* IFD: 10 Number of Directory Entries 15
* 12 NewSubfileType
* 24 ImageWidth Number of columns is a user parameter
* 36 ImageLength Number of rows is a user parameter
* 48 BitsPerSample 8, 8, 8
* 60 Compression Hard-coded no compression (for now)
* 72 PhotometricInterpretation Value is a user parameter
* 84 StripOffsets Offset and count determined as strips added
* 96 SamplesPerPixel Hard-coded to 3
* 108 RowsPerStrip Value is a user parameter
* 120 StripByteCounts Offset and count determined as strips added
* 132 XResolution Value is a user parameter
* 144 YResolution Value is a user parameter
* 156 Resolution Unit Hard-coded to "inches"
* 168 Software
* 180 DateTime
* 192 Next IFD offset 0
* 194 [2 bytes padding]
* Values:
* 196 XResolution Hard-coded to 300/1
* 204 YResolution Hard-coded to 300/1
* 212 BitsPerSample 8, 8, 8
* 218 [2 bytes padding]
* 220 "NuttX" Length = 6 (including NUL terminator)
* 226 "YYYY:MM:DD HH:MM:SS" Length = 20 (ncluding NUL terminator)
* 246 [2 bytes padding]
* 248 StripByteCounts Beginning of strip byte counts
* xxx StripOffsets Beginning of strip offsets
* xxx [Probably padding]
* xxx Data for strips Beginning of strip data
*/
#define TIFF_RGB_NIFDENTRIES 15
#define TIFF_RGB_STRIPIFDOFFS 84
#define TIFF_RGB_STRIPBCIFDOFFS 120
#define TIFF_RGB_VALOFFSET 196
#define TIFF_RGB_XRESOFFSET 196
#define TIFF_RGB_YRESOFFSET 204
#define TIFF_RGB_BPSOFFSET 212
#define TIFF_RGB_SWOFFSET 220
#define TIFF_RGB_DATEOFFSET 226
#define TIFF_RGB_STRIPBCOFFSET 248
/* Debug *******************************************************************/
/* CONFIG_DEBUG_TIFFOFFSETS may be defined (along with CONFIG_DEBUG_FEATURES and
* CONFIG_DEBUG_GRAPHICS) in order to verify the pre-determined TIFF file
* offsets.
*/
#if !defined(CONFIG_DEBUG_FEATURES) || !defined(CONFIG_DEBUG_GRAPHICS)
# undef CONFIG_DEBUG_TIFFOFFSETS
#endif
#ifdef CONFIG_DEBUG_TIFFOFFSETS
# define tiff_offset(o,l) (o) += (l)
# define tiff_checkoffs(o,x) DEBUGASSERT((o) == (x))
#else
# define tiff_offset(o,l)
# define tiff_checkoffs(o,x)
#endif
/****************************************************************************
* Private Types
****************************************************************************/
/****************************************************************************
* Private Data
****************************************************************************/
static const struct tiff_filefmt_s g_bilevinfo =
{
TIFF_BILEV_NIFDENTRIES, /* nifdentries, Number of IFD entries */
TIFF_BILEV_STRIPIFDOFFS, /* soifdoffset, Offset to StripOffset IFD entry */
TIFF_BILEV_STRIPBCIFDOFFS, /* sbcifdoffset, Offset to StripByteCount IFD entry */
TIFF_BILEV_VALOFFSET, /* valoffset, Offset to first values */
TIFF_BILEV_XRESOFFSET, /* xresoffset, Offset to XResolution values */
TIFF_BILEV_YRESOFFSET, /* yresoffset, Offset to yResolution values */
TIFF_BILEV_SWOFFSET, /* swoffset, Offset to Software string */
TIFF_BILEV_DATEOFFSET, /* dateoffset, Offset to DateTime string */
TIFF_BILEV_STRIPBCOFFSET /* sbcoffset, Offset to StripByteCount values */
};
static const struct tiff_filefmt_s g_greyinfo =
{
TIFF_GREY_NIFDENTRIES, /* nifdentries, Number of IFD entries */
TIFF_GREY_STRIPIFDOFFS, /* soifdoffset, Offset to StripOffset IFD entry */
TIFF_GREY_STRIPBCIFDOFFS, /* sbcifdoffset, Offset to StripByteCount IFD entry */
TIFF_GREY_VALOFFSET, /* valoffset, Offset to first values */
TIFF_GREY_XRESOFFSET, /* xresoffset, Offset to XResolution values */
TIFF_GREY_YRESOFFSET, /* yresoffset, Offset to yResolution values */
TIFF_GREY_SWOFFSET, /* swoffset, Offset to Software string */
TIFF_GREY_DATEOFFSET, /* dateoffset, Offset to DateTime string */
TIFF_GREY_STRIPBCOFFSET /* sbcoffset, Offset to StripByteCount values */
};
static const struct tiff_filefmt_s g_rgbinfo =
{
TIFF_RGB_NIFDENTRIES, /* nifdentries, Number of IFD entries */
TIFF_RGB_STRIPIFDOFFS, /* soifdoffset, Offset to StripOffset IFD entry */
TIFF_RGB_STRIPBCIFDOFFS, /* sbcifdoffset, Offset to StripByteCount IFD entry */
TIFF_RGB_VALOFFSET, /* valoffset, Offset to first values */
TIFF_RGB_XRESOFFSET, /* xresoffset, Offset to XResolution values */
TIFF_RGB_YRESOFFSET, /* yresoffset, Offset to yResolution values */
TIFF_RGB_SWOFFSET, /* swoffset, Offset to Software string */
TIFF_RGB_DATEOFFSET, /* dateoffset, Offset to DateTime string */
TIFF_RGB_STRIPBCOFFSET /* sbcoffset, Offset to StripByteCount values */
};
/****************************************************************************
* Public Data
****************************************************************************/
/****************************************************************************
* Private Functions
****************************************************************************/
/****************************************************************************
* Name: tiff_putheader
*
* Description:
* Setup to create a new TIFF file.
*
* Input Parameters:
* info - A pointer to the caller allocated parameter passing/TIFF state
* instance.
*
* Returned Value:
* Zero (OK) on success. A negated errno value on failure.
*
****************************************************************************/
static inline int tiff_putheader(FAR struct tiff_info_s *info)
{
struct tiff_header_s hdr;
int ret;
/* 0-1: Byte order */
#ifdef CONFIG_ENDIAN_BIG
hdr.order[0] = 'M'; /* "MM"=big endian */
hdr.order[1] = 'M';
#else
hdr.order[0] = 'I'; /* "II"=little endian */
hdr.order[1] = 'I';
#endif
/* 2-3: 42 in appropriate byte order */
tiff_put16(hdr.magic, 42);
/* 4-7: Offset to the first IFD */
tiff_put32(hdr.offset, TIFF_IFD_OFFSET);
/* Write the header to the output file */
ret = tiff_write(info->outfd, &hdr, SIZEOF_TIFF_HEADER);
if (ret != OK)
{
return ret;
}
/* Two pad bytes following the header */
ret = tiff_putint16(info->outfd, 0);
return ret;
}
/****************************************************************************
* Name: tiff_putifdentry
*
* Description:
* Write an IFD entry to outfile
*
* Input Parameters:
* info - A pointer to the caller allocated parameter passing/TIFF state
* instance.
* tag - The value for the IFD tag field
* type - The value for the IFD type field
* count - The value for the IFD count field
* offset - The value for the IFD offset field
*
* Returned Value:
* Zero (OK) on success. A negated errno value on failure.
*
****************************************************************************/
static int tiff_putifdentry(FAR struct tiff_info_s *info, uint16_t tag,
uint16_t type, uint32_t count, uint32_t offset)
{
struct tiff_ifdentry_s ifd;
tiff_put16(ifd.tag, tag);
tiff_put16(ifd.type, type);
tiff_put32(ifd.count, count);
tiff_put32(ifd.offset, offset);
return tiff_write(info->outfd, &ifd, SIZEOF_IFD_ENTRY);
}
/****************************************************************************
* Name: tiff_putifdentry
*
* Description:
* Write an IFD with a 16-bit immediate value
*
* Input Parameters:
* info - A pointer to the caller allocated parameter passing/TIFF state
* instance.
* tag - The value for the IFD tag field
* type - The value for the IFD type field
* count - The value for the IFD count field
* value - The 16-bit immediate value
*
* Returned Value:
* Zero (OK) on success. A negated errno value on failure.
*
****************************************************************************/
static int tiff_putifdentry16(FAR struct tiff_info_s *info, uint16_t tag,
uint16_t type, uint32_t count, uint16_t value)
{
union
{
uint8_t b[4];
uint32_t w;
} u;
u.w = 0;
tiff_put16(u.b, value);
return tiff_putifdentry(info, tag, type, count, u.w);
}
/****************************************************************************
* Name: tiff_datetime
*
* Description:
* Get the DateTime string
*
* Input Parameters:
*
*
* Returned Value:
* Zero (OK) on success. A negated errno value on failure.
*
****************************************************************************/
static int tiff_datetime(FAR char *timbuf, unsigned int buflen)
{
struct timespec ts;
struct tm tm;
int ret;
/* Get the current time */
ret = clock_gettime(CLOCK_REALTIME, &ts);
if (ret < 0)
{
gerr("ERROR: clock_gettime failed: %d\n", errno);
return ERROR;
}
/* Break the current time up into the format needed by strftime */
gmtime_r((FAR const time_t*)&ts.tv_sec, &tm);
/* Comvert the current time in the TIFF format */
strftime(timbuf, buflen, TIFF_DATETIME_FORMAT, &tm);
return OK;
}
/****************************************************************************
* Public Functions
****************************************************************************/
/****************************************************************************
* Name: tiff_initialize
*
* Description:
* Setup to create a new TIFF file.
*
* Input Parameters:
* info - A pointer to the caller allocated parameter passing/TIFF state instance.
*
* Returned Value:
* Zero (OK) on success. A negated errno value on failure.
*
****************************************************************************/
int tiff_initialize(FAR struct tiff_info_s *info)
{
uint16_t val16;
#ifdef CONFIG_DEBUG_TIFFOFFSETS
off_t offset = 0;
#endif
char timbuf[TIFF_DATETIME_STRLEN + 8];
int ret = -EINVAL;
DEBUGASSERT(info && info->outfile && info->tmpfile1 && info->tmpfile2);
/* Open all output files */
info->outfd = open(info->outfile, O_RDWR|O_CREAT|O_TRUNC, 0666);
if (info->outfd < 0)
{
gerr("ERROR: Failed to open %s for reading/writing: %d\n",
info->outfile, errno);
goto errout;
}
info->tmp1fd = open(info->tmpfile1, O_RDWR|O_CREAT|O_TRUNC, 0666);
if (info->tmp1fd < 0)
{
gerr("ERROR: Failed to open %s for reading/writing: %d\n",
info->tmpfile1, errno);
goto errout;
}
info->tmp2fd = open(info->tmpfile2, O_RDWR|O_CREAT|O_TRUNC, 0666);
if (info->tmp2fd < 0)
{
gerr("ERROR: Failed to open %s for reading/writing: %d\n",
info->tmpfile2, errno);
goto errout;
}
/* Make some decisions using the color format. Only the following are
* supported:
*/
info->pps = info->imgwidth * info->rps; /* Pixels per strip */
switch (info->colorfmt)
{
case FB_FMT_Y1: /* BPP=1, monochrome, 0=black */
info->filefmt = &g_bilevinfo; /* Bi-level file image file info */
info->imgflags = IMGFLAGS_FMT_Y1; /* Bit encoded image characteristics */
info->bps = (info->pps + 7) >> 3; /* Bytes per strip */
break;
case FB_FMT_Y4: /* BPP=4, 4-bit greyscale, 0=black */
info->filefmt = &g_greyinfo; /* Greyscale file image file info */
info->imgflags = IMGFLAGS_FMT_Y4; /* Bit encoded image characteristics */
info->bps = (info->pps + 1) >> 1; /* Bytes per strip */
break;
case FB_FMT_Y8: /* BPP=8, 8-bit greyscale, 0=black */
info->filefmt = &g_greyinfo; /* Greyscale file image file info */
info->imgflags = IMGFLAGS_FMT_Y8; /* Bit encoded image characteristics */
info->bps = info->pps; /* Bytes per strip */
break;
case FB_FMT_RGB16_565: /* BPP=16 R=6, G=6, B=5 */
info->filefmt = &g_rgbinfo; /* RGB file image file info */
info->imgflags = IMGFLAGS_FMT_RGB16_565; /* Bit encoded image characteristics */
info->bps = 3 * info->pps; /* Bytes per strip */
break;
case FB_FMT_RGB24: /* BPP=24 R=8, G=8, B=8 */
info->filefmt = &g_rgbinfo; /* RGB file image file info */
info->imgflags = IMGFLAGS_FMT_RGB24; /* Bit encoded image characteristics */
info->bps = 3 *info->pps; /* Bytes per strip */
break;
default:
gerr("ERROR: Unsupported color format: %d\n", info->colorfmt);
return -EINVAL;
}
/* Write the TIFF header data to the outfile:
*
* Header: 0 Byte Order "II" or "MM"
* 2 Magic Number 42
* 4 1st IFD offset 10
* 8 [2 bytes padding]
*/
ret = tiff_putheader(info);
if (ret < 0)
{
goto errout;
}
tiff_offset(offset, TIFF_IFD_OFFSET);
/* Write the Number of directory entries
*
* All formats: Offset 10 Number of Directory Entries 12
*/
ret = tiff_putint16(info->outfd, info->filefmt->nifdentries);
if (ret < 0)
{
goto errout;
}
tiff_offset(offset, 2);
/* Write the NewSubfileType IFD entry
*
* All formats: Offset 12 NewSubfileType
*/
ret = tiff_putifdentry16(info, IFD_TAG_NEWSUBFILETYPE, IFD_FIELD_LONG, 1, 0);
if (ret < 0)
{
goto errout;
}
tiff_offset(offset, SIZEOF_IFD_ENTRY);
/* Write ImageWidth and ImageLength
*
* All formats: Offset 24 ImageWidth Number of columns is a user parameter
* 36 ImageLength Number of rows is a user parameter
*/
ret = tiff_putifdentry16(info, IFD_TAG_IMAGEWIDTH, IFD_FIELD_SHORT, 1, info->imgwidth);
if (ret == OK)
{
ret= tiff_putifdentry16(info, IFD_TAG_IMAGELENGTH, IFD_FIELD_SHORT, 1, info->imgheight);
}
if (ret < 0)
{
goto errout;
}
tiff_offset(offset, 2*SIZEOF_IFD_ENTRY);
/* Write BitsPerSample
*
* Bi-level Images: None
* Greyscale: Offset 48 BitsPerSample (4 or 8)
* RGB: Offset 48 BitsPerSample (8,8,8)
*/
tiff_checkoffs(offset, 48);
if (IMGFLAGS_ISGREY(info->imgflags))
{
if (IMGFLAGS_ISGREY8(info->imgflags))
{
val16 = 8;
}
else
{
val16 = 4;
}
ret = tiff_putifdentry16(info, IFD_TAG_BITSPERSAMPLE, IFD_FIELD_SHORT, 1, val16);
if (ret < 0)
{
goto errout;
}
tiff_offset(offset, SIZEOF_IFD_ENTRY);
}
else if (IMGFLAGS_ISRGB(info->imgflags))
{
ret = tiff_putifdentry(info, IFD_TAG_BITSPERSAMPLE, IFD_FIELD_SHORT, 3, TIFF_RGB_BPSOFFSET);
if (ret < 0)
{
goto errout;
}
tiff_offset(offset, SIZEOF_IFD_ENTRY);
}
/* Write Compression:
*
* Bi-level Images: Offset 48 Hard-coded no compression (for now)
* Greyscale: Offset 60 " " " " "" " " " " " "
* RGB: Offset 60 " " " " "" " " " " " "
*/
ret = tiff_putifdentry16(info, IFD_TAG_COMPRESSION, IFD_FIELD_SHORT, 1, TAG_COMP_NONE);
if (ret < 0)
{
goto errout;
}
tiff_offset(offset, SIZEOF_IFD_ENTRY);
/* Write PhotometricInterpretation:
*
* Bi-level Images: Offset 48 Hard-coded BlackIsZero
* Greyscale: Offset 72 Hard-coded BlackIsZero
* RGB: Offset 72 Hard-coded RGB
*/
if (IMGFLAGS_ISRGB(info->imgflags))
{
val16 = TAG_PMI_RGB;
}
else
{
val16 = TAG_PMI_BLACK;
}
ret = tiff_putifdentry16(info, IFD_TAG_PMI, IFD_FIELD_SHORT, 1, val16);
if (ret < 0)
{
goto errout;
}
tiff_offset(offset, SIZEOF_IFD_ENTRY);
/* Write StripOffsets:
*
* Bi-level Images: Offset 72 Value determined by switch statement above
* Greyscale: Offset 84 Value determined by switch statement above
* RGB: Offset 84 Value determined by switch statement above
*/
tiff_checkoffs(offset, info->filefmt->soifdoffset);
ret = tiff_putifdentry(info, IFD_TAG_STRIPOFFSETS, IFD_FIELD_LONG, 0, 0);
if (ret < 0)
{
goto errout;
}
tiff_offset(offset, SIZEOF_IFD_ENTRY);
/* Write SamplesPerPixel
*
* Bi-level Images: N/A
* Greyscale: N/A
* RGB: Offset 96 Hard-coded to 3
*/
if (IMGFLAGS_ISRGB(info->imgflags))
{
ret = tiff_putifdentry16(info, IFD_TAG_SAMPLESPERPIXEL, IFD_FIELD_SHORT, 1, 3);
if (ret < 0)
{
goto errout;
}
tiff_offset(offset, SIZEOF_IFD_ENTRY);
}
/* Write RowsPerStrip:
*
* Bi-level Images: Offset 84 Value is a user parameter
* Greyscale: Offset 96 Value is a user parameter
* RGB: Offset 108 Value is a user parameter
*/
ret = tiff_putifdentry16(info, IFD_TAG_ROWSPERSTRIP, IFD_FIELD_SHORT, 1, info->rps);
if (ret < 0)
{
goto errout;
}
tiff_offset(offset, SIZEOF_IFD_ENTRY);
/* Write StripByteCounts:
*
* Bi-level Images: Offset 96 Count determined as strips added, Value offset = 216
* Greyscale: Offset 108 Count determined as strips added, Value offset = 228
* RGB: Offset 120 Count determined as strips added, Value offset = 248
*/
tiff_checkoffs(offset, info->filefmt->sbcifdoffset);
ret = tiff_putifdentry(info, IFD_TAG_STRIPCOUNTS, IFD_FIELD_LONG, 0, info->filefmt->sbcoffset);
if (ret < 0)
{
goto errout;
}
tiff_offset(offset, SIZEOF_IFD_ENTRY);
/* Write XResolution and YResolution:
*
* Bi-level Images: Offset 108 and 120, Values are a user parameters
* Greyscale: Offset 120 and 132, Values are a user parameters
* RGB: Offset 132 and 144, Values are a user parameters
*/
ret = tiff_putifdentry(info, IFD_TAG_XRESOLUTION, IFD_FIELD_RATIONAL, 1, info->filefmt->xresoffset);
if (ret == OK)
{
ret = tiff_putifdentry(info, IFD_TAG_YRESOLUTION, IFD_FIELD_RATIONAL, 1, info->filefmt->yresoffset);
}
if (ret < 0)
{
goto errout;
}
tiff_offset(offset, 2*SIZEOF_IFD_ENTRY);
/* Write ResolutionUnit:
*
* Bi-level Images: Offset 132, Hard-coded to "inches"
* Greyscale: Offset 144, Hard-coded to "inches"
* RGB: Offset 156, Hard-coded to "inches"
*/
ret = tiff_putifdentry16(info, IFD_TAG_RESUNIT, IFD_FIELD_SHORT, 1, TAG_RESUNIT_INCH);
if (ret < 0)
{
goto errout;
}
tiff_offset(offset, SIZEOF_IFD_ENTRY);
/* Write Software:
*
* Bi-level Images: Offset 144 Count, Hard-coded "NuttX"
* Greyscale: Offset 156 Count, Hard-coded "NuttX"
* RGB: Offset 168 Count, Hard-coded "NuttX"
*/
ret = tiff_putifdentry(info, IFD_TAG_SOFTWARE, IFD_FIELD_ASCII, TIFF_SOFTWARE_STRLEN, info->filefmt->swoffset);
if (ret < 0)
{
goto errout;
}
tiff_offset(offset, SIZEOF_IFD_ENTRY);
/* Write DateTime:
*
* Bi-level Images: Offset 156 Count, Format "YYYY:MM:DD HH:MM:SS"
* Greyscale: Offset 168 Count, Format "YYYY:MM:DD HH:MM:SS"
* RGB: Offset 180 Count, Format "YYYY:MM:DD HH:MM:SS"
*/
ret = tiff_putifdentry(info, IFD_TAG_DATETIME, IFD_FIELD_ASCII, TIFF_DATETIME_STRLEN, info->filefmt->dateoffset);
if (ret < 0)
{
goto errout;
}
tiff_offset(offset, SIZEOF_IFD_ENTRY);
/* Write Next IFD Offset and 2 bytes of padding:
*
* Bi-level Images: Offset 168, Next IFD offset
* Offset 170, [2 bytes padding]
* Greyscale: Offset 180, Next IFD offset
* Offset 182, [2 bytes padding]
* RGB: Offset 192, Next IFD offset
* Offset 194, [2 bytes padding]
*/
ret = tiff_putint32(info->outfd, 0);
if (ret < 0)
{
goto errout;
}
tiff_offset(offset, 4);
/* Now we begin the value section of the file */
tiff_checkoffs(offset, info->filefmt->valoffset);
/* Write the XResolution and YResolution data:
*
* Bi-level Images: Offset 172 Count, Hard-coded to 300/1
* Offset 180 Count, Hard-coded to 300/1
* Greyscale: Offset 184 Count, Hard-coded to 300/1
* Offset 192 Count, Hard-coded to 300/1
* RGB: Offset 196 Count, Hard-coded to 300/1
* Offset 204 Count, Hard-coded to 300/1
*/
tiff_checkoffs(offset, info->filefmt->xresoffset);
ret = tiff_putint32(info->outfd, 300);
if (ret == OK)
{
ret = tiff_putint32(info->outfd, 1);
}
if (ret < 0)
{
goto errout;
}
tiff_offset(offset, 8);
tiff_checkoffs(offset, info->filefmt->yresoffset);
ret = tiff_putint32(info->outfd, 300);
if (ret == OK)
{
ret = tiff_putint32(info->outfd, 1);
}
if (ret < 0)
{
goto errout;
}
tiff_offset(offset, 8);
/* Write RGB BitsPerSample Data:
*
* Bi-level Images: N/A
* Greyscale: N/A
* RGB: Offset 212 BitsPerSample (8,8,8)
* Offset 218 [2 bytes padding]
*/
if (IMGFLAGS_ISRGB(info->imgflags))
{
tiff_checkoffs(offset, TIFF_RGB_BPSOFFSET);
tiff_putint16(info->outfd, 8);
tiff_putint16(info->outfd, 8);
tiff_putint16(info->outfd, 8);
tiff_putint16(info->outfd, 0);
tiff_offset(offset, 8);
}
/* Write the Software string:
*
*
* Bi-level Images: Offset 188, Hard-coded "NuttX"
* Greyscale: Offset 200, Hard-coded "NuttX"
* RGB: Offset 220, Hard-coded "NuttX"
*/
tiff_checkoffs(offset, info->filefmt->swoffset);
ret = tiff_putstring(info->outfd, TIFF_SOFTWARE_STRING, TIFF_SOFTWARE_STRLEN);
if (ret < 0)
{
goto errout;
}
tiff_offset(offset, TIFF_SOFTWARE_STRLEN);
/* Write the DateTime string:
*
*
* Bi-level Images: Offset 188, Format "YYYY:MM:DD HH:MM:SSS"
* Greyscale: Offset 200, Hard-coded "NuttX"
* RGB: Offset 220, Hard-coded "NuttX"
*/
tiff_checkoffs(offset, info->filefmt->dateoffset);
ret = tiff_datetime(timbuf, TIFF_DATETIME_STRLEN + 8);
if (ret < 0)
{
goto errout;
}
ret = tiff_putstring(info->outfd, timbuf, TIFF_DATETIME_STRLEN);
if (ret < 0)
{
goto errout;
}
tiff_offset(offset, TIFF_DATETIME_STRLEN);
/* Add two bytes of padding */
ret = tiff_putint16(info->outfd, 0);
if (ret < 0)
{
goto errout;
}
tiff_offset(offset, 2);
/* And that should do it! */
tiff_checkoffs(offset, info->filefmt->sbcoffset);
info->outsize = info->filefmt->sbcoffset;
return OK;
errout:
tiff_abort(info);
return ret;
}