git migration: Refactor the ASTERISK_FILE_VERSION macro
[asterisk/asterisk.git] / formats / format_jpeg.c
1 /*
2  * Asterisk -- An open source telephony toolkit.
3  *
4  * Copyright (C) 1999 - 2005, Digium, Inc.
5  *
6  * Mark Spencer <markster@digium.com>
7  *
8  * See http://www.asterisk.org for more information about
9  * the Asterisk project. Please do not directly contact
10  * any of the maintainers of this project for assistance;
11  * the project provides a web site, mailing lists and IRC
12  * channels for your use.
13  *
14  * This program is free software, distributed under the terms of
15  * the GNU General Public License Version 2. See the LICENSE file
16  * at the top of the source tree.
17  */
18
19 /*! \file
20  *
21  * \brief JPEG File format support.
22  * 
23  * \arg File name extension: jpeg, jpg
24  * \ingroup formats
25  */
26
27 /*** MODULEINFO
28         <support_level>extended</support_level>
29  ***/
30  
31 #include "asterisk.h"
32
33 ASTERISK_REGISTER_FILE()
34
35 #include "asterisk/mod_format.h"
36 #include "asterisk/module.h"
37 #include "asterisk/image.h"
38 #include "asterisk/endian.h"
39 #include "asterisk/format_cache.h"
40
41 static struct ast_frame *jpeg_read_image(int fd, int len)
42 {
43         struct ast_frame fr;
44         int res;
45         char buf[65536];
46         if (len > sizeof(buf) || len < 0) {
47                 ast_log(LOG_WARNING, "JPEG image too large to read\n");
48                 return NULL;
49         }
50         res = read(fd, buf, len);
51         if (res < len) {
52                 ast_log(LOG_WARNING, "Only read %d of %d bytes: %s\n", res, len, strerror(errno));
53         }
54         memset(&fr, 0, sizeof(fr));
55         fr.frametype = AST_FRAME_IMAGE;
56         fr.subclass.format = ast_format_jpeg;
57         fr.data.ptr = buf;
58         fr.src = "JPEG Read";
59         fr.datalen = len;
60         return ast_frisolate(&fr);
61 }
62
63 static int jpeg_identify(int fd)
64 {
65         char buf[10];
66         int res;
67         res = read(fd, buf, sizeof(buf));
68         if (res < sizeof(buf))
69                 return 0;
70         if (memcmp(buf + 6, "JFIF", 4))
71                 return 0;
72         return 1;
73 }
74
75 static int jpeg_write_image(int fd, struct ast_frame *fr)
76 {
77         int res=0;
78         if (fr->datalen) {
79                 res = write(fd, fr->data.ptr, fr->datalen);
80                 if (res != fr->datalen) {
81                         ast_log(LOG_WARNING, "Only wrote %d of %d bytes: %s\n", res, fr->datalen, strerror(errno));
82                         return -1;
83                 }
84         }
85         return res;
86 }
87
88 static struct ast_imager jpeg_format = {
89         .name = "jpg",
90         .desc = "JPEG (Joint Picture Experts Group)",
91         .exts = "jpg|jpeg",
92         .read_image = jpeg_read_image,
93         .identify = jpeg_identify,
94         .write_image = jpeg_write_image,
95 };
96
97 static int load_module(void)
98 {
99         jpeg_format.format = ast_format_jpeg;
100         if (ast_image_register(&jpeg_format))
101                 return AST_MODULE_LOAD_FAILURE;
102         return AST_MODULE_LOAD_SUCCESS;
103 }
104
105 static int unload_module(void)
106 {
107         ast_image_unregister(&jpeg_format);
108
109         return 0;
110 }
111
112 AST_MODULE_INFO(ASTERISK_GPL_KEY, AST_MODFLAG_LOAD_ORDER, "jpeg (joint picture experts group) image format",
113         .support_level = AST_MODULE_SUPPORT_EXTENDED,
114         .load = load_module,
115         .unload = unload_module,
116         .load_pri = AST_MODPRI_APP_DEPEND
117 );