This repository has been archived by the owner on Sep 28, 2023. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 41
/
jpegloader.c
100 lines (76 loc) · 2.6 KB
/
jpegloader.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
#include "accimage.h"
#include <setjmp.h>
#include <jpeglib.h>
#ifndef LIBJPEG_TURBO_VERSION
#error libjpeg-turbo required (not IJG libjpeg)
#endif
struct accimage_jpeg_error_mgr {
struct jpeg_error_mgr pub;
jmp_buf setjmp_buffer;
};
static void accimage_jpeg_error_exit(j_common_ptr cinfo) {
struct accimage_jpeg_error_mgr* accimage_err = (struct accimage_jpeg_error_mgr*) cinfo->err;
longjmp(accimage_err->setjmp_buffer, 1);
}
void image_from_file(ImageObject* self, FILE* file) {
struct jpeg_decompress_struct state = { 0 };
struct accimage_jpeg_error_mgr jpeg_error;
unsigned char* buffer = NULL;
state.err = jpeg_std_error(&jpeg_error.pub);
jpeg_error.pub.error_exit = accimage_jpeg_error_exit;
if (setjmp(jpeg_error.setjmp_buffer)) {
char error_message[JMSG_LENGTH_MAX];
(*state.err->format_message)((j_common_ptr) &state, error_message);
PyErr_Format(PyExc_IOError, "JPEG decoding failed: %s", error_message);
goto cleanup;
}
jpeg_create_decompress(&state);
jpeg_stdio_src(&state, file);
jpeg_read_header(&state, TRUE);
state.dct_method = JDCT_ISLOW;
state.output_components = 3;
state.out_color_components = 3;
state.out_color_space = JCS_RGB;
jpeg_start_decompress(&state);
buffer = malloc(state.output_components * state.output_width * state.output_height);
if (buffer == NULL) {
PyErr_NoMemory();
goto cleanup;
}
while (state.output_scanline < state.output_height) {
unsigned char* row = buffer + state.output_scanline * state.output_width * state.output_components;
jpeg_read_scanlines(&state, &row, 1);
}
jpeg_finish_decompress(&state);
/* Success. Commit object state */
self->buffer = buffer;
buffer = NULL;
self->channels = 3;
self->height = state.output_height;
self->width = state.output_width;
self->row_stride = state.output_width;
self->y_offset = 0;
self->x_offset = 0;
cleanup:
jpeg_destroy_decompress(&state);
free(buffer);
if (file != NULL) {
fclose(file);
}
}
void image_from_buffer(ImageObject* self, void* buf, size_t size) {
FILE* source = fmemopen(buf, size, "rb");
if (source == NULL) {
PyErr_Format(PyExc_IOError, "failed to call fmemopen on buffer");
return;
}
image_from_file(self, source);
}
void image_from_jpeg(ImageObject* self, const char* path) {
FILE* file = file = fopen(path, "rb");
if (file == NULL) {
PyErr_Format(PyExc_IOError, "failed to open file %s", path);
return;
}
image_from_file(self, file);
}