Blob Blame Raw


#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <assert.h>

#include "toonz.h"
#include "tmsg.h"
#include "file.h"
#include "tiff.h"
#include "tiffio.h"
#include "version.h"
#include "history.h"
#include "colorsdb.h"
#include "ImageP/img_security.h"

#define LATEST_PLT_TYPE 4

static int Next_img_read_plt_without_buffer = FALSE;
static int Read_without_buffer              = FALSE;
#define SET_READ_WITHOUT_BUFFER                                                \
  {                                                                            \
    Read_without_buffer              = Next_img_read_plt_without_buffer;       \
    Next_img_read_plt_without_buffer = FALSE;                                  \
  }

#ifdef VECCHIA_MANIERA
static char *build_string_names(IMAGE *image);
static void set_color_names(IMAGE *image, char *names);
static char *plt_find_color_name(IMAGE *img, int color);
static char *plt_find_pencil_name(IMAGE *img, int color);
static void plt_set_color_name(IMAGE *img, int color, char *name);
static void plt_set_pencil_name(IMAGE *img, int pencil, char *name);
#endif

/*---------------------------------------------------------------------------*/

int img_write_plt(char *filename, IMAGE *image) {
  TIFF *tfp;
  UCHAR *buffer;
  int i, j, scanline, width, rows_per_strip;
  LPIXEL *cmap, *color, *pencil;
  USHORT palette[TOONZPALETTE_COUNT];
  char *names, *history;
  int plt_type, cmap_size;

  CHECK_IMAGEDLL_LICENSE_AND_GET_IMG_LICENSE_ATTR

  names = NIL;

  if (image->type != CMAP) return FALSE;

  tfp = TIFFOpen(filename, "w");
  if (!tfp) return FALSE;

  plt_type = (image->cmap.offset > 0) ? 4 : 3;

  if (!image->cmap.info.n_colors) {
    assert(plt_type == 4);
    image->cmap.info           = Tcm_old_default_info;
    image->cmap.info.n_colors  = image->cmap.color_n;
    image->cmap.info.n_pencils = image->cmap.pencil_n;
    cmap_size                  = TCM_MIN_CMAP_BUFFER_SIZE(image->cmap.info);
  } else {
    cmap_size = TCM_CMAP_BUFFER_SIZE(image->cmap.info);
  }
  palette[0]  = plt_type;
  palette[1]  = image->cmap.offset;
  palette[2]  = cmap_size <= 0xffff ? cmap_size : 0;
  palette[3]  = 0 /* image->cmap.info.tone_offs */;
  palette[4]  = image->cmap.info.tone_bits;
  palette[5]  = image->cmap.info.color_offs;
  palette[6]  = image->cmap.info.color_bits;
  palette[7]  = image->cmap.info.pencil_offs;
  palette[8]  = image->cmap.info.pencil_bits;
  palette[9]  = image->cmap.info.offset_mask;
  palette[10] = image->cmap.info.n_colors;
  palette[11] = image->cmap.info.n_pencils;

  for (i = 12; i < TOONZPALETTE_COUNT - 1; i++) palette[i] = 0;

  palette[TOONZPALETTE_COUNT - 1] = (Img_license_attr & TA_TOONZ_EDU) != 0;

  TIFFSetField(tfp, TIFFTAG_TOONZPALETTE, palette);

  if (plt_type == 3) {
    assert(image->cmap.color && image->cmap.pencil);
    width = image->cmap.color_n + image->cmap.pencil_n;
  } else {
    if (image->cmap.color && image->cmap.pencil)
      width = cmap_size + image->cmap.color_n + image->cmap.pencil_n;
    else
      width = cmap_size;
  }
  TIFFSetField(tfp, TIFFTAG_BITSPERSAMPLE, 8);
  TIFFSetField(tfp, TIFFTAG_SAMPLESPERPIXEL, 4);
  TIFFSetField(tfp, TIFFTAG_IMAGEWIDTH, width);
  TIFFSetField(tfp, TIFFTAG_IMAGELENGTH, 1);
  TIFFSetField(tfp, TIFFTAG_COMPRESSION, COMPRESSION_NONE);
  TIFFSetField(tfp, TIFFTAG_PLANARCONFIG, PLANARCONFIG_CONTIG);
  TIFFSetField(tfp, TIFFTAG_PHOTOMETRIC, PHOTOMETRIC_RGB);

#ifdef VECCHIA_MANIERA

  names_pointer = (TREE *)image->cmap.names;
  image->cmap.names =
      cdb_encode_names((TREE *)image->cmap.names, &(image->cmap.names_max));
  str                  = build_string_names(image);
  names_string_pointer = (char *)image->cmap.names;
  image->cmap.names    = (USHORT *)names_pointer;
  if (names_string_pointer) free(names_string_pointer);

#endif

  names = cdb_encode_all(image->cmap.names);

  TIFFSetField(tfp, TIFFTAG_TOONZCOLORNAMES, names);

  /* Aggiungo History */
  history = build_history();
  if (image->history) {
    switch (check_history(image->history, history)) {
      CASE APPEND : image->history = append_history(image->history, history);
      CASE REPLACE : image->history =
          replace_last_history(image->history, history);
    DEFAULT:
      printf("Internal error: bad history type; aborting...\n");
      abort();
    }
    free(history);
  } else
    image->history = history;

  TIFFSetField(tfp, TIFFTAG_TOONZHISTORY, image->history);

  scanline = TIFFScanlineSize(tfp);

  /*
* massima lunghezza di bytes in una strip e' 8k
* vedi Graphics File Formats pag.48
*/
  rows_per_strip = (8 * 1024) / scanline;

  TIFFSetField(tfp, TIFFTAG_ROWSPERSTRIP,
               rows_per_strip == 0 ? 1L : rows_per_strip);

  TMALLOC(buffer, scanline);
  cmap   = image->cmap.buffer;
  color  = image->cmap.color;
  pencil = image->cmap.pencil;

  switch (plt_type) {
    CASE 1
        : printf(
              "img_rwite_plt error: type 1 .plt's are not written any more\n");
    abort();
    CASE 2 : __OR 4 : for (i = 0, j = 0; i < cmap_size; i++, j += 4) {
      buffer[j + 0] = cmap[i].r;
      buffer[j + 1] = cmap[i].g;
      buffer[j + 2] = cmap[i].b;
      buffer[j + 3] = cmap[i].m;
    }
    if (width > cmap_size) {
      for (i = 0; i < image->cmap.color_n; i++, j += 4) {
        buffer[j + 0] = color[i].r;
        buffer[j + 1] = color[i].g;
        buffer[j + 2] = color[i].b;
        buffer[j + 3] = color[i].m;
      }
      for (i = 0; i < image->cmap.pencil_n; i++, j += 4) {
        buffer[j + 0] = pencil[i].r;
        buffer[j + 1] = pencil[i].g;
        buffer[j + 2] = pencil[i].b;
        buffer[j + 3] = pencil[i].m;
      }
    }
    CASE 3 : for (i = 0, j = 0; i < image->cmap.color_n; i++, j += 4) {
      buffer[j + 0] = color[i].m; /* different from type 2 */
      buffer[j + 1] = color[i].b;
      buffer[j + 2] = color[i].g;
      buffer[j + 3] = color[i].r;
    }
    for (i = 0; i < image->cmap.pencil_n; i++, j += 4) {
      buffer[j + 0] = pencil[i].m;
      buffer[j + 1] = pencil[i].b;
      buffer[j + 2] = pencil[i].g;
      buffer[j + 3] = pencil[i].r;
    }
  DEFAULT:
    assert(!"bad palette type");
  }

  if (TIFFWriteScanline(tfp, buffer, 0, 0) < 0) goto bad;
  TFREE(buffer);
  TIFFClose(tfp);
  TFREE(names);
  return TRUE;

bad:
  TFREE(buffer);
  TIFFClose(tfp);
  TFREE(names);
  return FALSE;
}

/*===========================================================================*/

void next_img_read_plt_without_buffer(void) {
  Next_img_read_plt_without_buffer = TRUE;
}

/*---------------------------------------------------------------------------*/

IMAGE *img_read_plt(char *filename) {
  int with_cmap_buffer;
  IMAGE *image = NIL;
  TIFF *tfp;
  UCHAR *buffer = NIL;
  int i, j, scanline;
  LPIXEL *cmap;
  USHORT *palette; /*  [TOONZPALETTE_COUNT] */
  USHORT spp, bps, comp, plan_con, photom;
  int width, height;
  char *names;
  int plt_type, m, cmap_file_size, cmap_alloc_size;
  int colpen_cmap, colbuf_alloc_size, penbuf_alloc_size;
  int max_n_colors, max_n_pencils;
  int act_n_colors, act_n_pencils, index;
  LPIXEL grey, black;
  TBOOL edu_file;

  CHECK_IMAGEDLL_LICENSE_AND_GET_IMG_LICENSE_ATTR

  SET_READ_WITHOUT_BUFFER

  tfp = TIFFOpen(filename, "r");
  if (!tfp) return NIL;

  TIFFGetField(tfp, TIFFTAG_TOONZPALETTE, &palette);
  TIFFGetField(tfp, TIFFTAG_TOONZCOLORNAMES, &names);

  TIFFGetField(tfp, TIFFTAG_BITSPERSAMPLE, &bps);
  TIFFGetField(tfp, TIFFTAG_SAMPLESPERPIXEL, &spp);
  TIFFGetField(tfp, TIFFTAG_IMAGEWIDTH, &width);
  TIFFGetField(tfp, TIFFTAG_IMAGELENGTH, &height);
  TIFFGetField(tfp, TIFFTAG_COMPRESSION, &comp);
  TIFFGetField(tfp, TIFFTAG_PLANARCONFIG, &plan_con);
  TIFFGetField(tfp, TIFFTAG_PHOTOMETRIC, &photom);

  if (photom != PHOTOMETRIC_RGB || plan_con != PLANARCONFIG_CONTIG) goto bad;

  edu_file = palette[TOONZPALETTE_COUNT - 1] & 1;
  if (edu_file && !(Img_license_attr & TA_TOONZ_EDU)) {
    char str[1024];
    BUILD_EDU_ERROR_STRING(str)
    tmsg_error(str);
    goto bad;
  }

  scanline = TIFFScanlineSize(tfp);

  image = new_img();
  if (!image) goto bad;
  image->type = CMAP;

  /* Leggo history */
  if (!TIFFGetField(tfp, TIFFTAG_TOONZHISTORY, &image->history))
    image->history = "";
  image->history   = strsave(image->history);

  plt_type             = palette[0];
  image->cmap.offset   = palette[1];
  image->cmap.color_n  = palette[10];
  image->cmap.pencil_n = palette[11];

  image->cmap.info = Tcm_old_default_info;
  if (palette[6]) {
    /*image->cmap.info.tone_offs   = palette[3]; */
    image->cmap.info.tone_bits   = palette[4];
    image->cmap.info.color_offs  = palette[5];
    image->cmap.info.color_bits  = palette[6];
    image->cmap.info.pencil_offs = palette[7];
    image->cmap.info.pencil_bits = palette[8];
    image->cmap.info.offset_mask = palette[9];
    image->cmap.info.n_tones     = 1 << palette[4];
    image->cmap.info.n_colors    = palette[10];
    image->cmap.info.n_pencils   = palette[11];
  }
  image->cmap.info.default_val =
      (image->cmap.info.n_tones - 1) | image->cmap.info.offset_mask;

  cmap_alloc_size   = TCM_CMAP_BUFFER_SIZE(image->cmap.info);
  cmap_file_size    = palette[2] ? palette[2] : cmap_alloc_size;
  colpen_cmap       = image->cmap.info.n_tones > 16;
  colbuf_alloc_size = TCM_CMAP_COLBUFFER_SIZE(image->cmap.info);
  penbuf_alloc_size = TCM_CMAP_PENBUFFER_SIZE(image->cmap.info);

  TMALLOC(buffer, scanline);
  if (!buffer) goto bad;

  with_cmap_buffer = !Read_without_buffer || (plt_type <= 2) || (plt_type == 4);
  if (with_cmap_buffer) {
    if (colpen_cmap) {
      TMALLOC(image->cmap.colbuffer, colbuf_alloc_size);
      TMALLOC(image->cmap.penbuffer, penbuf_alloc_size);
      if (!image->cmap.colbuffer || !image->cmap.penbuffer) goto bad;
    } else {
      TMALLOC(image->cmap.buffer, cmap_alloc_size);
      if (!image->cmap.buffer) goto bad;
    }
  }
  TCALLOC(image->cmap.color, 1 << image->cmap.info.color_bits);
  if (!image->cmap.color) goto bad;
  TCALLOC(image->cmap.pencil, 1 << image->cmap.info.pencil_bits);
  if (!image->cmap.pencil) goto bad;

  if (TIFFReadScanline(tfp, buffer, 0, 0) < 0) goto bad;

  switch (plt_type) {
    CASE 1 : if (!with_cmap_buffer) {
      TCALLOC(image->cmap.buffer, cmap_alloc_size)
      if (!image->cmap.buffer) goto bad;
    }
    for (i = 0, j = 0; i < cmap_file_size; i++, j += 4) {
      image->cmap.buffer[i].r = buffer[j + 0];
      image->cmap.buffer[i].g = buffer[j + 1];
      image->cmap.buffer[i].b = buffer[j + 2];
    }
    to_new_cmap(&image->cmap);
    set_colors_and_pencils(&image->cmap);
    if (!with_cmap_buffer) TFREE(image->cmap.buffer)

    CASE 2 : __OR 4
             : if (with_cmap_buffer) for (i = 0, j = 0; i < cmap_file_size;
                                          i++, j += 4) {
      image->cmap.buffer[i].r = buffer[j + 0];
      image->cmap.buffer[i].g = buffer[j + 1];
      image->cmap.buffer[i].b = buffer[j + 2];
      image->cmap.buffer[i].m = buffer[j + 3];
    }
    else j = 4 * cmap_file_size;

    if (width > cmap_file_size) {
      for (i = 0; i < image->cmap.color_n; i++, j += 4) {
        image->cmap.color[i].r = buffer[j + 0];
        image->cmap.color[i].g = buffer[j + 1];
        image->cmap.color[i].b = buffer[j + 2];
        image->cmap.color[i].m = buffer[j + 3];
      }
      for (i = 0; i < image->cmap.pencil_n; i++, j += 4) {
        image->cmap.pencil[i].r = buffer[j + 0];
        image->cmap.pencil[i].g = buffer[j + 1];
        image->cmap.pencil[i].b = buffer[j + 2];
        image->cmap.pencil[i].m = buffer[j + 3];
      }
    } else
      set_colors_and_pencils(&image->cmap);

    CASE 3 : for (i = 0, j = 0; i < image->cmap.color_n; i++, j += 4) {
      image->cmap.color[i].m = buffer[j + 0]; /* different from type 2 */
      image->cmap.color[i].b = buffer[j + 1];
      image->cmap.color[i].g = buffer[j + 2];
      image->cmap.color[i].r = buffer[j + 3];
    }
    for (i = 0; i < image->cmap.pencil_n; i++, j += 4) {
      image->cmap.pencil[i].m = buffer[j + 0];
      image->cmap.pencil[i].b = buffer[j + 1];
      image->cmap.pencil[i].g = buffer[j + 2];
      image->cmap.pencil[i].r = buffer[j + 3];
    }
    if (with_cmap_buffer)
      if (colpen_cmap) {
        memset(image->cmap.colbuffer, 0, colbuf_alloc_size * sizeof(LPIXEL));
        memset(image->cmap.penbuffer, 0, penbuf_alloc_size * sizeof(LPIXEL));
        fill_cmap_colbuffer(image->cmap.colbuffer, image->cmap.info,
                            image->cmap.color, FALSE);
        fill_cmap_penbuffer(image->cmap.penbuffer, image->cmap.info,
                            image->cmap.pencil, FALSE);
      } else {
        memset(image->cmap.buffer, 0, cmap_alloc_size * sizeof(LPIXEL));
        fill_cmap_buffer(image->cmap.buffer, image->cmap.info,
                         image->cmap.color, image->cmap.pencil, FALSE);
      }
  DEFAULT:
    goto bad;
  }

#ifdef VECCHIA_MANIERA

  set_color_names(image, names);
  names_pointer = image->cmap.names;
  image->cmap.names =
      (USHORT *)cdb_decode_names(image->cmap.names, image->cmap.names_max);
  TFREE(names_pointer);

#endif

  image->cmap.names = cdb_decode_all(names, image->cmap.info);

  /* estendo la palette */

  max_n_colors  = 1 << image->cmap.info.color_bits;
  max_n_pencils = 1 << image->cmap.info.pencil_bits;
  if (max_n_colors > image->cmap.info.n_colors ||
      max_n_pencils > image->cmap.info.n_pencils) {
    act_n_colors               = image->cmap.info.n_colors;
    act_n_pencils              = image->cmap.info.n_pencils;
    image->cmap.info.n_colors  = max_n_colors;
    image->cmap.info.n_pencils = max_n_pencils;
    image->cmap.color_n        = image->cmap.info.n_colors;
    image->cmap.pencil_n       = image->cmap.info.n_pencils;
    grey.r = grey.g = grey.b = 127;
    grey.m                   = 255;
    black.r = black.g = black.b = 0;
    black.m                     = 255;
    for (i = act_n_colors; i < max_n_colors; i++) {
      image->cmap.color[i] = grey;
      index                = TCM_COLOR_INDEX(image->cmap.info, i);
      cdb_set_group(image, index, UNUSED_COLOR_PAGE_NAME);
      if (with_cmap_buffer && colpen_cmap)
        fill_cmap_colramp(image->cmap.colbuffer, image->cmap.info,
                          image->cmap.color[i], i, FALSE);
    }
    for (i = act_n_pencils; i < max_n_pencils; i++) {
      image->cmap.pencil[i] = black;
      index                 = TCM_PENCIL_INDEX(image->cmap.info, i);
      cdb_set_group(image, index, UNUSED_COLOR_PAGE_NAME);
      if (with_cmap_buffer && colpen_cmap)
        fill_cmap_penramp(image->cmap.penbuffer, image->cmap.info,
                          image->cmap.pencil[i], i, FALSE);
    }
    if (with_cmap_buffer && !colpen_cmap)
      fill_cmap_buffer(image->cmap.buffer, image->cmap.info, image->cmap.color,
                       image->cmap.pencil, FALSE);
  }

  if (buffer) TFREE(buffer);
  TIFFClose(tfp);
  return image;

bad:
  if (image) free_img(image);
  TFREE(buffer);
  TIFFClose(tfp);
  return NIL;
}

/*---------------------------------------------------------------------------*/

IMAGE *img_read_plt_info(char *filename) {
  IMAGE *image = NIL;
  TIFF *tfp;
  USHORT *palette; /*  [TOONZPALETTE_COUNT] */
  USHORT plan_con, photom;
  char *names;
  int max_n_colors, max_n_pencils;
  TBOOL edu_file;

  CHECK_IMAGEDLL_LICENSE_AND_GET_IMG_LICENSE_ATTR

  tfp = TIFFOpen(filename, "r");
  if (!tfp) return NIL;

  TIFFGetField(tfp, TIFFTAG_TOONZPALETTE, &palette);
  TIFFGetField(tfp, TIFFTAG_TOONZCOLORNAMES, &names);
  TIFFGetField(tfp, TIFFTAG_PLANARCONFIG, &plan_con);
  TIFFGetField(tfp, TIFFTAG_PHOTOMETRIC, &photom);

  if (photom != PHOTOMETRIC_RGB || plan_con != PLANARCONFIG_CONTIG) goto bad;

  edu_file = palette[TOONZPALETTE_COUNT - 1] & 1;
  if (edu_file && !(Img_license_attr & TA_TOONZ_EDU)) {
    char str[1024];
    BUILD_EDU_ERROR_STRING(str)
    tmsg_warning(str);
  }

  image = new_img();
  if (!image) goto bad;
  image->type = CMAP;

  if (palette[0] < 1 || palette[0] > LATEST_PLT_TYPE) goto bad;

  image->cmap.offset   = palette[1];
  image->cmap.color_n  = palette[10];
  image->cmap.pencil_n = palette[11];

  image->cmap.info = Tcm_old_default_info;
  if (palette[6]) {
    /*image->cmap.info.tone_offs   = palette[3]; */
    image->cmap.info.tone_bits   = palette[4];
    image->cmap.info.color_offs  = palette[5];
    image->cmap.info.color_bits  = palette[6];
    image->cmap.info.pencil_offs = palette[7];
    image->cmap.info.pencil_bits = palette[8];
    image->cmap.info.offset_mask = palette[9];
    image->cmap.info.n_tones     = 1 << palette[4];
    image->cmap.info.n_colors    = palette[10];
    image->cmap.info.n_pencils   = palette[11];
  }
  image->cmap.info.default_val =
      (image->cmap.info.n_tones - 1) | image->cmap.info.offset_mask;

  /* Leggo history */
  if (!TIFFGetField(tfp, TIFFTAG_TOONZHISTORY, &image->history))
    image->history = "";
  image->history   = strsave(image->history);

  /* estendo la palette */

  max_n_colors  = 1 << image->cmap.info.color_bits;
  max_n_pencils = 1 << image->cmap.info.pencil_bits;
  if (max_n_colors > image->cmap.info.n_colors ||
      max_n_pencils > image->cmap.info.n_pencils) {
    image->cmap.info.n_colors  = max_n_colors;
    image->cmap.info.n_pencils = max_n_pencils;
    image->cmap.color_n        = image->cmap.info.n_colors;
    image->cmap.pencil_n       = image->cmap.info.n_pencils;
  }

  image->cmap.names = cdb_decode_all(names, image->cmap.info);

  TIFFClose(tfp);
  return image;

bad:
  if (image) free_img(image);
  TIFFClose(tfp);
  return NIL;
}

/*---------------------------------------------------------------------------*/
/* Attenzione: se si modifica questa funzione, occore riportare le
 *             variazioni nella libreria line x line (simg_plt.c)
 */

#ifdef VECCHIA_MANIERA

static char *plt_find_color_name(IMAGE *img, int color) {
  char *name;
  int index;

  index = TCM_COLOR_INDEX(img->cmap.info, color);
  name  = find_color_name(img, index);

  return name;
}

#endif

/*---------------------------------------------------------------------------*/
/* Attenzione: se si modifica questa funzione, occore riportare le
 *             variazioni nella libreria line x line (simg_plt.c)
 */

#ifdef VECCHIA_MANIERA

static char *plt_find_pencil_name(IMAGE *img, int pencil) {
  char *name;
  int index;

  index = TCM_PENCIL_INDEX(img->cmap.info, pencil);
  name  = find_color_name(img, index);

  return name;
}

#endif

/*---------------------------------------------------------------------------*/
/* Attenzione: se si modifica questa funzione, occore riportare le
 *             variazioni nella libreria line x line (simg_plt.c)
 */

#ifdef VECCHIA_MANIERA

static void plt_set_color_name(IMAGE *img, int color, char *name) {
  int index;

  index = TCM_COLOR_INDEX(img->cmap.info, color);
  set_color_name(img, index, name);
}

#endif

/*---------------------------------------------------------------------------*/
/* Attenzione: se si modifica questa funzione, occore riportare le
 *             variazioni nella libreria line x line (simg_plt.c)
 */

#ifdef VECCHIA_MANIERA

static void plt_set_pencil_name(IMAGE *img, int pencil, char *name) {
  int index;

  index = TCM_PENCIL_INDEX(img->cmap.info, pencil);
  set_color_name(img, index, name);
}

#endif

/*---------------------------------------------------------------------------*/
/* Attenzione: se si modifica questa funzione, occore riportare le
 *             variazioni nella libreria line x line (simg_plt.c)
 */

#ifdef VECCHIA_MANIERA

static char *build_string_names(IMAGE *image) {
  char *str, *name;
  int i, maxsize = 1000, ptr, len;

  str = (char *)malloc(maxsize);
  ptr = 0;

  for (i = 0; i < image->cmap.color_n; i++) {
    name = plt_find_color_name(image, i);
    if (name && *name) {
      len = strlen(name);
      if ((len + ptr) >= maxsize) {
        maxsize += 200;
        str = (char *)realloc(str, maxsize);
      }
      memmove(str + ptr, name, len);
      ptr += len;
    }
    if (ptr + 1 >= maxsize) {
      maxsize += 200;
      str = (char *)realloc(str, maxsize);
    }
    memmove(str + ptr, "|", 1);
    ptr++;
  }

  for (i = 0; i < image->cmap.pencil_n; i++) {
    name = plt_find_pencil_name(image, i);
    if (name && *name) {
      len = strlen(name);
      if ((len + ptr) >= maxsize) {
        maxsize += 200;
        str = (char *)realloc(str, maxsize);
      }
      memmove(str + ptr, name, len);
      ptr += len;
    }
    if (ptr + 1 >= maxsize) {
      maxsize += 200;
      str = (char *)realloc(str, maxsize);
    }
    memmove(str + ptr, "|", 1);
    ptr++;
  }

  str[ptr] = 0;
  return str;
}

#endif

/*---------------------------------------------------------------------------*/
/* Attenzione: se si modifica questa funzione, occore riportare le
 *             variazioni nella libreria line x line (simg_plt.c)
 */

#ifdef VECCHIA_MANIERA

static void set_color_names(IMAGE *image, char *names) {
  int i;
  char *s, *name;

  s = names;
  i = 0;
  while (*s) {
    name = NULL;
    if (*s != '|') {
      name = s;
      while (*s && *s != '|') s++;
      *s = 0;
    }

    if (name) {
      if (i < image->cmap.info.n_colors)
        plt_set_color_name(image, i, name);
      else
        plt_set_pencil_name(image, i - image->cmap.info.n_colors, name);
    }
    i++;
    s++;
  }
}

#endif