mirror of https://github.com/GNOME/gimp.git
340 lines
7.3 KiB
C
340 lines
7.3 KiB
C
/* GIMP - The GNU Image Manipulation Program
|
|
* Copyright (C) 1995 Spencer Kimball and Peter Mattis
|
|
*
|
|
* This program is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 3 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include <string.h>
|
|
|
|
#include <gio/gio.h>
|
|
|
|
#include "core/core-types.h"
|
|
|
|
#include "xcf-private.h"
|
|
#include "xcf-write.h"
|
|
|
|
#include "gimp-intl.h"
|
|
|
|
|
|
guint
|
|
xcf_write_int8 (XcfInfo *info,
|
|
const guint8 *data,
|
|
gint count,
|
|
GError **error)
|
|
{
|
|
GError *my_error = NULL;
|
|
gsize bytes_written = 0;
|
|
|
|
/* we allow for 'data == NULL && count == 0', which
|
|
* g_output_stream_write_all() rejects.
|
|
*/
|
|
if (count > 0)
|
|
{
|
|
if (! g_output_stream_write_all (info->output, data, count,
|
|
&bytes_written, NULL, &my_error))
|
|
{
|
|
g_propagate_prefixed_error (error, my_error,
|
|
_("Error writing XCF: "));
|
|
}
|
|
|
|
info->cp += bytes_written;
|
|
}
|
|
|
|
return bytes_written;
|
|
}
|
|
|
|
guint
|
|
xcf_write_int16 (XcfInfo *info,
|
|
const guint16 *data,
|
|
gint count,
|
|
GError **error)
|
|
{
|
|
GError *tmp_error = NULL;
|
|
gint i;
|
|
|
|
if (count > 0)
|
|
{
|
|
for (i = 0; i < count; i++)
|
|
{
|
|
guint16 tmp = g_htons (data[i]);
|
|
|
|
xcf_write_int8 (info, (const guint8 *) &tmp, 2, &tmp_error);
|
|
|
|
if (tmp_error)
|
|
{
|
|
g_propagate_error (error, tmp_error);
|
|
|
|
return i * 2;
|
|
}
|
|
}
|
|
}
|
|
|
|
return count * 2;
|
|
}
|
|
|
|
guint
|
|
xcf_write_int32 (XcfInfo *info,
|
|
const guint32 *data,
|
|
gint count,
|
|
GError **error)
|
|
{
|
|
GError *tmp_error = NULL;
|
|
gint i;
|
|
|
|
if (count > 0)
|
|
{
|
|
for (i = 0; i < count; i++)
|
|
{
|
|
guint32 tmp = g_htonl (data[i]);
|
|
|
|
xcf_write_int8 (info, (const guint8 *) &tmp, 4, &tmp_error);
|
|
|
|
if (tmp_error)
|
|
{
|
|
g_propagate_error (error, tmp_error);
|
|
|
|
return i * 4;
|
|
}
|
|
}
|
|
}
|
|
|
|
return count * 4;
|
|
}
|
|
|
|
guint
|
|
xcf_write_int64 (XcfInfo *info,
|
|
const guint64 *data,
|
|
gint count,
|
|
GError **error)
|
|
{
|
|
GError *tmp_error = NULL;
|
|
gint i;
|
|
|
|
if (count > 0)
|
|
{
|
|
for (i = 0; i < count; i++)
|
|
{
|
|
guint64 tmp = GINT64_TO_BE (data[i]);
|
|
|
|
xcf_write_int8 (info, (const guint8 *) &tmp, 8, &tmp_error);
|
|
|
|
if (tmp_error)
|
|
{
|
|
g_propagate_error (error, tmp_error);
|
|
|
|
return i * 8;
|
|
}
|
|
}
|
|
}
|
|
|
|
return count * 8;
|
|
}
|
|
|
|
guint
|
|
xcf_write_offset (XcfInfo *info,
|
|
const goffset *data,
|
|
gint count,
|
|
GError **error)
|
|
{
|
|
if (count > 0)
|
|
{
|
|
gint i;
|
|
|
|
for (i = 0; i < count; i++)
|
|
{
|
|
GError *tmp_error = NULL;
|
|
|
|
if (info->bytes_per_offset == 4)
|
|
{
|
|
guint32 tmp = g_htonl (data[i]);
|
|
|
|
xcf_write_int8 (info, (const guint8 *) &tmp, 4, &tmp_error);
|
|
}
|
|
else
|
|
{
|
|
gint64 tmp = GINT64_TO_BE (data[i]);
|
|
|
|
xcf_write_int8 (info, (const guint8 *) &tmp, 8, &tmp_error);
|
|
}
|
|
|
|
if (tmp_error)
|
|
{
|
|
g_propagate_error (error, tmp_error);
|
|
|
|
return i * info->bytes_per_offset;
|
|
}
|
|
}
|
|
}
|
|
|
|
return count * info->bytes_per_offset;
|
|
}
|
|
|
|
guint
|
|
xcf_write_zero_offset (XcfInfo *info,
|
|
gint count,
|
|
GError **error)
|
|
{
|
|
if (count > 0)
|
|
{
|
|
guint8 *tmp;
|
|
guint bytes_written = 0;
|
|
|
|
tmp = g_try_malloc (count * info->bytes_per_offset);
|
|
if (! tmp)
|
|
{
|
|
g_set_error (error, G_FILE_ERROR, G_FILE_ERROR_FAILED,
|
|
_("Error writing XCF: failed to allocate %d bytes of memory."),
|
|
count * info->bytes_per_offset);
|
|
}
|
|
else
|
|
{
|
|
memset (tmp, 0, count * info->bytes_per_offset);
|
|
|
|
bytes_written = xcf_write_int8 (info, (const guint8 *) tmp,
|
|
count * info->bytes_per_offset, error);
|
|
g_free (tmp);
|
|
}
|
|
|
|
return bytes_written;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
guint
|
|
xcf_write_float (XcfInfo *info,
|
|
const gfloat *data,
|
|
gint count,
|
|
GError **error)
|
|
{
|
|
return xcf_write_int32 (info,
|
|
(const guint32 *)((gconstpointer) data), count,
|
|
error);
|
|
}
|
|
|
|
guint
|
|
xcf_write_string (XcfInfo *info,
|
|
gchar **data,
|
|
gint count,
|
|
GError **error)
|
|
{
|
|
GError *tmp_error = NULL;
|
|
guint total = 0;
|
|
gint i;
|
|
|
|
for (i = 0; i < count; i++)
|
|
{
|
|
guint32 tmp;
|
|
|
|
if (data[i])
|
|
tmp = strlen (data[i]) + 1;
|
|
else
|
|
tmp = 0;
|
|
|
|
xcf_write_int32 (info, &tmp, 1, &tmp_error);
|
|
|
|
if (tmp_error)
|
|
{
|
|
g_propagate_error (error, tmp_error);
|
|
return total;
|
|
}
|
|
|
|
if (tmp > 0)
|
|
xcf_write_int8 (info, (const guint8 *) data[i], tmp, &tmp_error);
|
|
|
|
if (tmp_error)
|
|
{
|
|
g_propagate_error (error, tmp_error);
|
|
return total;
|
|
}
|
|
|
|
total += 4 + tmp;
|
|
}
|
|
|
|
return total;
|
|
}
|
|
|
|
guint
|
|
xcf_write_component (XcfInfo *info,
|
|
gint bpc,
|
|
const guint8 *data,
|
|
gint count,
|
|
GError **error)
|
|
{
|
|
switch (bpc)
|
|
{
|
|
case 1:
|
|
return xcf_write_int8 (info, data, count, error);
|
|
|
|
case 2:
|
|
return xcf_write_int16 (info, (const guint16 *) data, count, error);
|
|
|
|
case 4:
|
|
return xcf_write_int32 (info, (const guint32 *) data, count, error);
|
|
|
|
case 8:
|
|
return xcf_write_int64 (info, (const guint64 *) data, count, error);
|
|
|
|
default:
|
|
g_set_error (error, G_FILE_ERROR, G_FILE_ERROR_FAILED,
|
|
_("Error writing XCF: unsupported BPC when writing pixel: %d"),
|
|
bpc);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
void
|
|
xcf_write_to_be (gint bpc,
|
|
guint8 *data,
|
|
gint count)
|
|
{
|
|
gint i;
|
|
|
|
switch (bpc)
|
|
{
|
|
case 1:
|
|
break;
|
|
|
|
case 2:
|
|
{
|
|
guint16 *d = (guint16 *) data;
|
|
|
|
for (i = 0; i < count; i++)
|
|
d[i] = g_htons (d[i]);
|
|
}
|
|
break;
|
|
|
|
case 4:
|
|
{
|
|
guint32 *d = (guint32 *) data;
|
|
|
|
for (i = 0; i < count; i++)
|
|
d[i] = g_htonl (d[i]);
|
|
}
|
|
break;
|
|
|
|
case 8:
|
|
{
|
|
guint64 *d = (guint64 *) data;
|
|
|
|
for (i = 0; i < count; i++)
|
|
d[i] = GINT64_TO_BE (d[i]);
|
|
}
|
|
break;
|
|
}
|
|
}
|