pdf-colorspace.c 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655
  1. // Copyright (C) 2004-2025 Artifex Software, Inc.
  2. //
  3. // This file is part of MuPDF.
  4. //
  5. // MuPDF is free software: you can redistribute it and/or modify it under the
  6. // terms of the GNU Affero General Public License as published by the Free
  7. // Software Foundation, either version 3 of the License, or (at your option)
  8. // any later version.
  9. //
  10. // MuPDF is distributed in the hope that it will be useful, but WITHOUT ANY
  11. // WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
  12. // FOR A PARTICULAR PURPOSE. See the GNU Affero General Public License for more
  13. // details.
  14. //
  15. // You should have received a copy of the GNU Affero General Public License
  16. // along with MuPDF. If not, see <https://www.gnu.org/licenses/agpl-3.0.en.html>
  17. //
  18. // Alternative licensing terms are available from the licensor.
  19. // For commercial licensing, see <https://www.artifex.com/> or contact
  20. // Artifex Software, Inc., 39 Mesa Street, Suite 108A, San Francisco,
  21. // CA 94129, USA, for further information.
  22. #include "mupdf/fitz.h"
  23. #include "mupdf/pdf.h"
  24. #include <string.h>
  25. static fz_colorspace *pdf_load_colorspace_imp(fz_context *ctx, pdf_obj *obj, pdf_cycle_list *cycle_up);
  26. /* ICCBased */
  27. static fz_colorspace *
  28. load_icc_based(fz_context *ctx, pdf_obj *dict, int allow_alt, pdf_cycle_list *cycle_up)
  29. {
  30. int n = pdf_dict_get_int(ctx, dict, PDF_NAME(N));
  31. fz_colorspace *alt = NULL;
  32. fz_colorspace *cs = NULL;
  33. pdf_obj *obj;
  34. fz_var(alt);
  35. fz_var(cs);
  36. /* Look at Alternate to detect type (especially Lab). */
  37. if (allow_alt)
  38. {
  39. obj = pdf_dict_get(ctx, dict, PDF_NAME(Alternate));
  40. if (obj)
  41. {
  42. fz_try(ctx)
  43. alt = pdf_load_colorspace_imp(ctx, obj, cycle_up);
  44. fz_catch(ctx)
  45. {
  46. fz_rethrow_if(ctx, FZ_ERROR_TRYLATER);
  47. fz_rethrow_if(ctx, FZ_ERROR_SYSTEM);
  48. fz_report_error(ctx);
  49. fz_warn(ctx, "ignoring broken ICC Alternate colorspace");
  50. }
  51. }
  52. }
  53. #if FZ_ENABLE_ICC
  54. {
  55. fz_buffer *buf = NULL;
  56. fz_var(buf);
  57. fz_try(ctx)
  58. {
  59. buf = pdf_load_stream(ctx, dict);
  60. cs = fz_new_icc_colorspace(ctx, alt ? alt->type : FZ_COLORSPACE_NONE, 0, NULL, buf);
  61. if (cs->n > n)
  62. {
  63. fz_warn(ctx, "ICC colorspace N=%d does not match profile N=%d (ignoring profile)", n, cs->n);
  64. fz_drop_colorspace(ctx, cs);
  65. cs = NULL;
  66. }
  67. else if (cs->n < n)
  68. {
  69. fz_warn(ctx, "ICC colorspace N=%d does not match profile N=%d (using profile)", n, cs->n);
  70. }
  71. }
  72. fz_always(ctx)
  73. fz_drop_buffer(ctx, buf);
  74. fz_catch(ctx)
  75. {
  76. if (fz_caught(ctx) == FZ_ERROR_TRYLATER || fz_caught(ctx) == FZ_ERROR_SYSTEM)
  77. {
  78. fz_drop_colorspace(ctx, alt);
  79. fz_rethrow(ctx);
  80. }
  81. fz_report_error(ctx);
  82. fz_warn(ctx, "ignoring broken ICC profile");
  83. }
  84. }
  85. #endif
  86. if (!cs)
  87. cs = alt;
  88. else
  89. fz_drop_colorspace(ctx, alt);
  90. if (!cs)
  91. {
  92. if (n == 1) cs = fz_keep_colorspace(ctx, fz_device_gray(ctx));
  93. else if (n == 3) cs = fz_keep_colorspace(ctx, fz_device_rgb(ctx));
  94. else if (n == 4) cs = fz_keep_colorspace(ctx, fz_device_cmyk(ctx));
  95. else fz_throw(ctx, FZ_ERROR_SYNTAX, "invalid ICC colorspace");
  96. }
  97. return cs;
  98. }
  99. static void
  100. devicen_eval(fz_context *ctx, void *tint, const float *sv, int sn, float *dv, int dn)
  101. {
  102. pdf_eval_function(ctx, tint, sv, sn, dv, dn);
  103. }
  104. static void
  105. devicen_drop(fz_context *ctx, void *tint)
  106. {
  107. pdf_drop_function(ctx, tint);
  108. }
  109. static fz_colorspace *
  110. load_devicen(fz_context *ctx, pdf_obj *array, int is_devn, pdf_cycle_list *cycle_up)
  111. {
  112. fz_colorspace *base = NULL;
  113. fz_colorspace *cs = NULL;
  114. pdf_obj *nameobj = pdf_array_get(ctx, array, 1);
  115. pdf_obj *baseobj = pdf_array_get(ctx, array, 2);
  116. pdf_obj *tintobj = pdf_array_get(ctx, array, 3);
  117. char name[100];
  118. int i, n;
  119. fz_var(cs);
  120. if (pdf_is_array(ctx, nameobj))
  121. {
  122. n = pdf_array_len(ctx, nameobj);
  123. if (n < 1)
  124. fz_throw(ctx, FZ_ERROR_SYNTAX, "too few components in DeviceN colorspace");
  125. if (n > FZ_MAX_COLORS)
  126. fz_throw(ctx, FZ_ERROR_SYNTAX, "too many components in DeviceN colorspace");
  127. }
  128. else
  129. {
  130. n = 1;
  131. }
  132. base = pdf_load_colorspace_imp(ctx, baseobj, cycle_up);
  133. fz_try(ctx)
  134. {
  135. if (is_devn)
  136. {
  137. fz_snprintf(name, sizeof name, "DeviceN(%d,%s", n, base->name);
  138. for (i = 0; i < n; i++) {
  139. fz_strlcat(name, ",", sizeof name);
  140. fz_strlcat(name, pdf_array_get_name(ctx, nameobj, i), sizeof name);
  141. }
  142. fz_strlcat(name, ")", sizeof name);
  143. }
  144. else
  145. {
  146. fz_snprintf(name, sizeof name, "Separation(%s,%s)", base->name, pdf_to_name(ctx, nameobj));
  147. }
  148. cs = fz_new_colorspace(ctx, FZ_COLORSPACE_SEPARATION, 0, n, name);
  149. cs->u.separation.eval = devicen_eval;
  150. cs->u.separation.drop = devicen_drop;
  151. cs->u.separation.base = fz_keep_colorspace(ctx, base);
  152. cs->u.separation.tint = pdf_load_function(ctx, tintobj, n, cs->u.separation.base->n);
  153. if (pdf_is_array(ctx, nameobj))
  154. for (i = 0; i < n; i++)
  155. fz_colorspace_name_colorant(ctx, cs, i, pdf_array_get_name(ctx, nameobj, i));
  156. else
  157. fz_colorspace_name_colorant(ctx, cs, 0, pdf_to_name(ctx, nameobj));
  158. }
  159. fz_always(ctx)
  160. {
  161. fz_drop_colorspace(ctx, base);
  162. }
  163. fz_catch(ctx)
  164. {
  165. fz_drop_colorspace(ctx, cs);
  166. fz_rethrow(ctx);
  167. }
  168. return cs;
  169. }
  170. int
  171. pdf_is_tint_colorspace(fz_context *ctx, fz_colorspace *cs)
  172. {
  173. return cs && cs->type == FZ_COLORSPACE_SEPARATION;
  174. }
  175. /* Indexed */
  176. static fz_colorspace *
  177. load_indexed(fz_context *ctx, pdf_obj *array, pdf_cycle_list *cycle_up)
  178. {
  179. pdf_obj *baseobj = pdf_array_get(ctx, array, 1);
  180. pdf_obj *highobj = pdf_array_get(ctx, array, 2);
  181. pdf_obj *lookupobj = pdf_array_get(ctx, array, 3);
  182. fz_colorspace *base = NULL;
  183. fz_colorspace *cs;
  184. size_t i, n;
  185. int high;
  186. unsigned char *lookup = NULL;
  187. fz_var(base);
  188. fz_var(lookup);
  189. fz_try(ctx)
  190. {
  191. base = pdf_load_colorspace_imp(ctx, baseobj, cycle_up);
  192. high = pdf_to_int(ctx, highobj);
  193. high = fz_clampi(high, 0, 255);
  194. n = (size_t)base->n * (high + 1);
  195. lookup = Memento_label(fz_malloc(ctx, n), "cs_lookup");
  196. if (pdf_is_string(ctx, lookupobj))
  197. {
  198. size_t sn = fz_minz(n, pdf_to_str_len(ctx, lookupobj));
  199. unsigned char *buf = (unsigned char *) pdf_to_str_buf(ctx, lookupobj);
  200. for (i = 0; i < sn; ++i)
  201. lookup[i] = buf[i];
  202. for (; i < n; ++i)
  203. lookup[i] = 0;
  204. }
  205. else if (pdf_is_indirect(ctx, lookupobj))
  206. {
  207. fz_stream *file = NULL;
  208. fz_var(file);
  209. fz_try(ctx)
  210. {
  211. file = pdf_open_stream(ctx, lookupobj);
  212. i = fz_read(ctx, file, lookup, n);
  213. if (i < n)
  214. memset(lookup+i, 0, n-i);
  215. }
  216. fz_always(ctx)
  217. {
  218. fz_drop_stream(ctx, file);
  219. }
  220. fz_catch(ctx)
  221. {
  222. fz_rethrow(ctx);
  223. }
  224. }
  225. else
  226. {
  227. fz_throw(ctx, FZ_ERROR_SYNTAX, "cannot parse colorspace lookup table");
  228. }
  229. cs = fz_new_indexed_colorspace(ctx, base, high, lookup);
  230. }
  231. fz_always(ctx)
  232. fz_drop_colorspace(ctx, base);
  233. fz_catch(ctx)
  234. {
  235. fz_free(ctx, lookup);
  236. fz_rethrow(ctx);
  237. }
  238. return cs;
  239. }
  240. static void
  241. pdf_load_cal_common(fz_context *ctx, pdf_obj *dict, float *wp, float *bp, float *gamma)
  242. {
  243. pdf_obj *obj;
  244. int i;
  245. obj = pdf_dict_get(ctx, dict, PDF_NAME(WhitePoint));
  246. if (pdf_array_len(ctx, obj) != 3)
  247. fz_throw(ctx, FZ_ERROR_SYNTAX, "WhitePoint must be a 3-element array");
  248. for (i = 0; i < 3; i++)
  249. {
  250. wp[i] = pdf_array_get_real(ctx, obj, i);
  251. if (wp[i] < 0)
  252. fz_throw(ctx, FZ_ERROR_SYNTAX, "WhitePoint numbers must be positive");
  253. }
  254. if (wp[1] != 1)
  255. fz_throw(ctx, FZ_ERROR_SYNTAX, "WhitePoint Yw must be 1.0");
  256. obj = pdf_dict_get(ctx, dict, PDF_NAME(BlackPoint));
  257. if (pdf_array_len(ctx, obj) == 3)
  258. {
  259. for (i = 0; i < 3; i++)
  260. {
  261. bp[i] = pdf_array_get_real(ctx, obj, i);
  262. if (bp[i] < 0)
  263. fz_throw(ctx, FZ_ERROR_SYNTAX, "BlackPoint numbers must be positive");
  264. }
  265. }
  266. obj = pdf_dict_get(ctx, dict, PDF_NAME(Gamma));
  267. if (pdf_is_number(ctx, obj))
  268. {
  269. gamma[0] = pdf_to_real(ctx, obj);
  270. gamma[1] = gamma[2];
  271. if (gamma[0] <= 0)
  272. fz_throw(ctx, FZ_ERROR_SYNTAX, "Gamma must be greater than zero");
  273. }
  274. else if (pdf_array_len(ctx, obj) == 3)
  275. {
  276. for (i = 0; i < 3; i++)
  277. {
  278. gamma[i] = pdf_array_get_real(ctx, obj, i);
  279. if (gamma[i] <= 0)
  280. fz_throw(ctx, FZ_ERROR_SYNTAX, "Gamma must be greater than zero");
  281. }
  282. }
  283. }
  284. static fz_colorspace *
  285. pdf_load_cal_gray(fz_context *ctx, pdf_obj *dict)
  286. {
  287. float wp[3];
  288. float bp[3] = { 0, 0, 0 };
  289. float gamma[3] = { 1, 1, 1 };
  290. if (dict == NULL)
  291. return fz_keep_colorspace(ctx, fz_device_gray(ctx));
  292. fz_try(ctx)
  293. pdf_load_cal_common(ctx, dict, wp, bp, gamma);
  294. fz_catch(ctx)
  295. return fz_keep_colorspace(ctx, fz_device_gray(ctx));
  296. return fz_new_cal_gray_colorspace(ctx, wp, bp, gamma[0]);
  297. }
  298. static fz_colorspace *
  299. pdf_load_cal_rgb(fz_context *ctx, pdf_obj *dict)
  300. {
  301. pdf_obj *obj;
  302. float matrix[9] = { 1, 0, 0, 0, 1, 0, 0, 0, 1 };
  303. float wp[3];
  304. float bp[3] = { 0, 0, 0 };
  305. float gamma[3] = { 1, 1, 1 };
  306. int i;
  307. if (dict == NULL)
  308. return fz_keep_colorspace(ctx, fz_device_rgb(ctx));
  309. fz_try(ctx)
  310. {
  311. pdf_load_cal_common(ctx, dict, wp, bp, gamma);
  312. obj = pdf_dict_get(ctx, dict, PDF_NAME(Matrix));
  313. if (pdf_array_len(ctx, obj) == 9)
  314. {
  315. for (i = 0; i < 9; i++)
  316. matrix[i] = pdf_array_get_real(ctx, obj, i);
  317. }
  318. }
  319. fz_catch(ctx)
  320. return fz_keep_colorspace(ctx, fz_device_rgb(ctx));
  321. return fz_new_cal_rgb_colorspace(ctx, wp, bp, gamma, matrix);
  322. }
  323. /* Parse and create colorspace from PDF object */
  324. static fz_colorspace *
  325. pdf_load_colorspace_imp(fz_context *ctx, pdf_obj *obj, pdf_cycle_list *cycle_up)
  326. {
  327. fz_colorspace *cs = NULL;
  328. pdf_cycle_list cycle;
  329. if (pdf_cycle(ctx, &cycle, cycle_up, obj))
  330. fz_throw(ctx, FZ_ERROR_SYNTAX, "recursive colorspace");
  331. if (pdf_is_name(ctx, obj))
  332. {
  333. if (pdf_name_eq(ctx, obj, PDF_NAME(Pattern)))
  334. return fz_keep_colorspace(ctx, fz_device_gray(ctx));
  335. else if (pdf_name_eq(ctx, obj, PDF_NAME(G)))
  336. return fz_keep_colorspace(ctx, fz_device_gray(ctx));
  337. else if (pdf_name_eq(ctx, obj, PDF_NAME(RGB)))
  338. return fz_keep_colorspace(ctx, fz_device_rgb(ctx));
  339. else if (pdf_name_eq(ctx, obj, PDF_NAME(CMYK)))
  340. return fz_keep_colorspace(ctx, fz_device_cmyk(ctx));
  341. else if (pdf_name_eq(ctx, obj, PDF_NAME(DeviceGray)))
  342. return fz_keep_colorspace(ctx, fz_device_gray(ctx));
  343. else if (pdf_name_eq(ctx, obj, PDF_NAME(DeviceRGB)))
  344. return fz_keep_colorspace(ctx, fz_device_rgb(ctx));
  345. else if (pdf_name_eq(ctx, obj, PDF_NAME(DeviceCMYK)))
  346. return fz_keep_colorspace(ctx, fz_device_cmyk(ctx));
  347. else
  348. fz_throw(ctx, FZ_ERROR_SYNTAX, "unknown colorspace: %s", pdf_to_name(ctx, obj));
  349. }
  350. else if (pdf_is_array(ctx, obj))
  351. {
  352. pdf_obj *name = pdf_array_get(ctx, obj, 0);
  353. if (pdf_is_name(ctx, name))
  354. {
  355. /* load base colorspace instead */
  356. if (pdf_name_eq(ctx, name, PDF_NAME(G)))
  357. return fz_keep_colorspace(ctx, fz_device_gray(ctx));
  358. else if (pdf_name_eq(ctx, name, PDF_NAME(RGB)))
  359. return fz_keep_colorspace(ctx, fz_device_rgb(ctx));
  360. else if (pdf_name_eq(ctx, name, PDF_NAME(CMYK)))
  361. return fz_keep_colorspace(ctx, fz_device_cmyk(ctx));
  362. else if (pdf_name_eq(ctx, name, PDF_NAME(DeviceGray)))
  363. return fz_keep_colorspace(ctx, fz_device_gray(ctx));
  364. else if (pdf_name_eq(ctx, name, PDF_NAME(DeviceRGB)))
  365. return fz_keep_colorspace(ctx, fz_device_rgb(ctx));
  366. else if (pdf_name_eq(ctx, name, PDF_NAME(DeviceCMYK)))
  367. return fz_keep_colorspace(ctx, fz_device_cmyk(ctx));
  368. else if (pdf_name_eq(ctx, name, PDF_NAME(CalCMYK)))
  369. return fz_keep_colorspace(ctx, fz_device_cmyk(ctx));
  370. else if (pdf_name_eq(ctx, name, PDF_NAME(Lab)))
  371. return fz_keep_colorspace(ctx, fz_device_lab(ctx));
  372. else
  373. {
  374. if ((cs = pdf_find_item(ctx, fz_drop_colorspace_imp, obj)) != NULL)
  375. return cs;
  376. if (pdf_name_eq(ctx, name, PDF_NAME(ICCBased)))
  377. cs = load_icc_based(ctx, pdf_array_get(ctx, obj, 1), 1, &cycle);
  378. else if (pdf_name_eq(ctx, name, PDF_NAME(CalGray)))
  379. cs = pdf_load_cal_gray(ctx, pdf_array_get(ctx, obj, 1));
  380. else if (pdf_name_eq(ctx, name, PDF_NAME(CalRGB)))
  381. cs = pdf_load_cal_rgb(ctx, pdf_array_get(ctx, obj, 1));
  382. else if (pdf_name_eq(ctx, name, PDF_NAME(Indexed)))
  383. cs = load_indexed(ctx, obj, &cycle);
  384. else if (pdf_name_eq(ctx, name, PDF_NAME(I)))
  385. cs = load_indexed(ctx, obj, &cycle);
  386. else if (pdf_name_eq(ctx, name, PDF_NAME(Separation)))
  387. cs = load_devicen(ctx, obj, 0, &cycle);
  388. else if (pdf_name_eq(ctx, name, PDF_NAME(DeviceN)))
  389. cs = load_devicen(ctx, obj, 1, &cycle);
  390. else if (pdf_name_eq(ctx, name, PDF_NAME(Pattern)))
  391. {
  392. pdf_obj *pobj = pdf_array_get(ctx, obj, 1);
  393. if (!pobj)
  394. return fz_keep_colorspace(ctx, fz_device_gray(ctx));
  395. cs = pdf_load_colorspace_imp(ctx, pobj, &cycle);
  396. }
  397. else
  398. fz_throw(ctx, FZ_ERROR_SYNTAX, "unknown colorspace %s", pdf_to_name(ctx, name));
  399. pdf_store_item(ctx, obj, cs, 1000);
  400. return cs;
  401. }
  402. }
  403. }
  404. /* We have seen files where /DefaultRGB is specified as 1 0 R,
  405. * and 1 0 obj << /Length 3144 /Alternate /DeviceRGB /N 3 >>
  406. * stream ...iccprofile... endstream endobj.
  407. * This *should* be [ /ICCBased 1 0 R ], but Acrobat seems to
  408. * handle it, so do our best. */
  409. else if (pdf_is_dict(ctx, obj))
  410. {
  411. if ((cs = pdf_find_item(ctx, fz_drop_colorspace_imp, obj)) != NULL)
  412. return cs;
  413. cs = load_icc_based(ctx, obj, 1, &cycle);
  414. pdf_store_item(ctx, obj, cs, 1000);
  415. return cs;
  416. }
  417. fz_throw(ctx, FZ_ERROR_SYNTAX, "could not parse color space (%d 0 R)", pdf_to_num(ctx, obj));
  418. }
  419. fz_colorspace *
  420. pdf_load_colorspace(fz_context *ctx, pdf_obj *obj)
  421. {
  422. return pdf_load_colorspace_imp(ctx, obj, NULL);
  423. }
  424. #if FZ_ENABLE_ICC
  425. static fz_colorspace *
  426. pdf_load_output_intent(fz_context *ctx, pdf_document *doc)
  427. {
  428. pdf_obj *root = pdf_dict_get(ctx, pdf_trailer(ctx, doc), PDF_NAME(Root));
  429. pdf_obj *intents = pdf_dict_get(ctx, root, PDF_NAME(OutputIntents));
  430. pdf_obj *intent_dict;
  431. pdf_obj *dest_profile;
  432. fz_colorspace *cs = NULL;
  433. /* An array of intents */
  434. if (!intents)
  435. return NULL;
  436. /* For now, always just use the first intent. I have never even seen a file
  437. * with multiple intents but it could happen */
  438. intent_dict = pdf_array_get(ctx, intents, 0);
  439. if (!intent_dict)
  440. return NULL;
  441. dest_profile = pdf_dict_get(ctx, intent_dict, PDF_NAME(DestOutputProfile));
  442. if (!dest_profile)
  443. return NULL;
  444. fz_var(cs);
  445. fz_try(ctx)
  446. cs = load_icc_based(ctx, dest_profile, 0, NULL);
  447. fz_catch(ctx)
  448. {
  449. fz_rethrow_if(ctx, FZ_ERROR_TRYLATER);
  450. fz_rethrow_if(ctx, FZ_ERROR_SYSTEM);
  451. fz_report_error(ctx);
  452. fz_warn(ctx, "Attempt to read Output Intent failed");
  453. }
  454. return cs;
  455. }
  456. fz_colorspace *
  457. pdf_document_output_intent(fz_context *ctx, pdf_document *doc)
  458. {
  459. if (!doc->oi)
  460. doc->oi = pdf_load_output_intent(ctx, doc);
  461. return doc->oi;
  462. }
  463. #else
  464. fz_colorspace *
  465. pdf_document_output_intent(fz_context *ctx, pdf_document *doc)
  466. {
  467. return NULL;
  468. }
  469. #endif
  470. static pdf_obj *
  471. pdf_add_indexed_colorspace(fz_context *ctx, pdf_document *doc, fz_colorspace *cs)
  472. {
  473. fz_colorspace *basecs;
  474. unsigned char *lookup = NULL;
  475. int high = 0;
  476. int basen;
  477. pdf_obj *obj;
  478. basecs = cs->u.indexed.base;
  479. high = cs->u.indexed.high;
  480. lookup = cs->u.indexed.lookup;
  481. basen = basecs->n;
  482. if (fz_colorspace_is_indexed(ctx, basecs))
  483. fz_throw(ctx, FZ_ERROR_FORMAT, "indexed colorspaces must not have an indexed colorspace as base");
  484. obj = pdf_add_new_array(ctx, doc, 4);
  485. fz_try(ctx)
  486. {
  487. pdf_array_push(ctx, obj, PDF_NAME(Indexed));
  488. pdf_array_push(ctx, obj, pdf_add_colorspace(ctx, doc, basecs));
  489. pdf_array_push_int(ctx, obj, high);
  490. pdf_array_push_string(ctx, obj, (char *) lookup, (size_t)basen * (high + 1));
  491. }
  492. fz_catch(ctx)
  493. {
  494. pdf_drop_obj(ctx, obj);
  495. fz_rethrow(ctx);
  496. }
  497. return obj;
  498. }
  499. static pdf_obj *
  500. pdf_add_icc_colorspace(fz_context *ctx, pdf_document *doc, fz_colorspace *cs)
  501. {
  502. #if FZ_ENABLE_ICC
  503. pdf_obj *obj = NULL;
  504. pdf_obj *stm = NULL;
  505. pdf_obj *base;
  506. pdf_obj *ref;
  507. pdf_colorspace_resource_key key;
  508. fz_var(obj);
  509. fz_var(stm);
  510. obj = pdf_find_colorspace_resource(ctx, doc, cs, &key);
  511. if (obj)
  512. return obj;
  513. switch (fz_colorspace_type(ctx, cs))
  514. {
  515. default:
  516. fz_throw(ctx, FZ_ERROR_UNSUPPORTED, "only Gray, RGB, and CMYK ICC colorspaces supported");
  517. case FZ_COLORSPACE_GRAY:
  518. base = PDF_NAME(DeviceGray);
  519. break;
  520. case FZ_COLORSPACE_RGB:
  521. base = PDF_NAME(DeviceRGB);
  522. break;
  523. case FZ_COLORSPACE_CMYK:
  524. base = PDF_NAME(DeviceCMYK);
  525. break;
  526. case FZ_COLORSPACE_LAB:
  527. base = PDF_NAME(Lab);
  528. break;
  529. }
  530. fz_try(ctx)
  531. {
  532. stm = pdf_add_stream(ctx, doc, cs->u.icc.buffer, NULL, 0);
  533. pdf_dict_put_int(ctx, stm, PDF_NAME(N), cs->n);
  534. pdf_dict_put(ctx, stm, PDF_NAME(Alternate), base);
  535. obj = pdf_add_new_array(ctx, doc, 2);
  536. pdf_array_push(ctx, obj, PDF_NAME(ICCBased));
  537. pdf_array_push(ctx, obj, stm);
  538. ref = pdf_insert_colorspace_resource(ctx, doc, &key, obj);
  539. }
  540. fz_always(ctx)
  541. {
  542. pdf_drop_obj(ctx, obj);
  543. pdf_drop_obj(ctx, stm);
  544. }
  545. fz_catch(ctx)
  546. fz_rethrow(ctx);
  547. return ref;
  548. #else
  549. fz_throw(ctx, FZ_ERROR_UNSUPPORTED, "ICC support disabled");
  550. #endif
  551. }
  552. pdf_obj *
  553. pdf_add_colorspace(fz_context *ctx, pdf_document *doc, fz_colorspace *cs)
  554. {
  555. if (fz_colorspace_is_indexed(ctx, cs))
  556. return pdf_add_indexed_colorspace(ctx, doc, cs);
  557. if (fz_colorspace_is_icc(ctx, cs))
  558. return pdf_add_icc_colorspace(ctx, doc, cs);
  559. switch (fz_colorspace_type(ctx, cs))
  560. {
  561. case FZ_COLORSPACE_NONE:
  562. case FZ_COLORSPACE_GRAY:
  563. return PDF_NAME(DeviceGray);
  564. case FZ_COLORSPACE_RGB:
  565. return PDF_NAME(DeviceRGB);
  566. case FZ_COLORSPACE_CMYK:
  567. return PDF_NAME(DeviceCMYK);
  568. case FZ_COLORSPACE_LAB:
  569. return PDF_NAME(Lab);
  570. default:
  571. fz_throw(ctx, FZ_ERROR_ARGUMENT, "only Gray, RGB, and CMYK colorspaces supported");
  572. break;
  573. }
  574. }