frei0r/factory.c, frei0r_helper.c: add support for color parameter type with whitespa...
[melted] / src / modules / frei0r / factory.c
1 /*
2 * factory.c -- the factory method interfaces
3 * Copyright (c) 2008 Marco Gittler <g.marco@freenet.de>
4 *
5 * This library is free software; you can redistribute it and/or
6 * modify it under the terms of the GNU Lesser General Public
7 * License as published by the Free Software Foundation; either
8 * version 2.1 of the License, or (at your option) any later version.
9 *
10 * This library is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13 * Lesser General Public License for more details.
14 *
15 * You should have received a copy of the GNU Lesser General Public
16 * License along with this library; if not, write to the Free Software
17 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
18 */
19
20 #include <string.h>
21 #include <framework/mlt.h>
22 #include <frei0r.h>
23
24 #include <stddef.h>
25 #include <stdio.h>
26 #include <sys/stat.h>
27 #include <sys/types.h>
28 #include <dirent.h>
29 #include <dlfcn.h>
30 #include <stdlib.h>
31 #include <limits.h>
32
33 #define FREI0R_PLUGIN_PATH "/usr/lib/frei0r-1:/usr/local/lib/frei0r-1:/opt/local/lib/frei0r-1"
34
35 extern mlt_filter filter_frei0r_init( mlt_profile profile, mlt_service_type type, const char *id, char *arg );
36 extern mlt_frame filter_process( mlt_filter this, mlt_frame frame );
37 extern void filter_close( mlt_filter this );
38 extern void transition_close( mlt_transition this );
39 extern mlt_frame transition_process( mlt_transition transition, mlt_frame a_frame, mlt_frame b_frame );
40
41 static mlt_properties fill_param_info ( mlt_service_type type, const char *service_name, char *name )
42 {
43 char file[ PATH_MAX ];
44 char servicetype[ 1024 ]="";
45 struct stat stat_buff;
46
47 switch ( type ) {
48 case filter_type:
49 strcpy ( servicetype , "filter" );
50 break;
51 case transition_type:
52 strcpy ( servicetype , "transition" ) ;
53 break;
54 default:
55 strcpy ( servicetype , "" );
56 };
57
58 snprintf( file, PATH_MAX, "%s/frei0r/%s_%s.yml", mlt_environment( "MLT_DATA" ), servicetype, service_name );
59 stat(file,&stat_buff);
60
61 if (S_ISREG(stat_buff.st_mode)){
62 return mlt_properties_parse_yaml( file );
63 }
64
65 void* handle=dlopen(name,RTLD_LAZY);
66 if (!handle) return NULL;
67 void (*plginfo)(f0r_plugin_info_t*)=dlsym(handle,"f0r_get_plugin_info");
68 void (*param_info)(f0r_param_info_t*,int param_index)=dlsym(handle,"f0r_get_param_info");
69 if (!plginfo || !param_info) {
70 dlclose(handle);
71 return NULL;
72 }
73 mlt_properties metadata = mlt_properties_new();
74 f0r_plugin_info_t info;
75 char string[48];
76 int j=0;
77
78 plginfo(&info);
79 snprintf ( string, sizeof(string) , "%d.%d" , info.major_version , info.minor_version );
80 mlt_properties_set ( metadata, "schema_version" , "0.1" );
81 mlt_properties_set ( metadata, "title" , info.name );
82 mlt_properties_set ( metadata, "version", string );
83 mlt_properties_set ( metadata, "identifier" , service_name );
84 mlt_properties_set ( metadata, "description" , info.explanation );
85 mlt_properties_set ( metadata, "creator" , info.author );
86 switch (type){
87 case filter_type:
88 mlt_properties_set ( metadata, "type" , "filter" );
89 break;
90 case transition_type:
91 mlt_properties_set ( metadata, "type" , "transition" );
92 break;
93 default:
94 break;
95 }
96
97 mlt_properties parameter = mlt_properties_new ( );
98 mlt_properties_set_data ( metadata , "parameters" , parameter , 0 , ( mlt_destructor )mlt_properties_close, NULL );
99 mlt_properties tags = mlt_properties_new ( );
100 mlt_properties_set_data ( metadata , "tags" , tags , 0 , ( mlt_destructor )mlt_properties_close, NULL );
101 mlt_properties_set ( tags , "0" , "Video" );
102
103 for (j=0;j<info.num_params;j++){
104 snprintf ( string , sizeof(string), "%d" , j );
105 mlt_properties pnum = mlt_properties_new ( );
106 mlt_properties_set_data ( parameter , string , pnum , 0 , ( mlt_destructor )mlt_properties_close, NULL );
107 f0r_param_info_t paraminfo;
108 param_info(&paraminfo,j);
109 mlt_properties_set ( pnum , "identifier" , paraminfo.name );
110 mlt_properties_set ( pnum , "title" , paraminfo.name );
111 mlt_properties_set ( pnum , "description" , paraminfo.explanation);
112 if ( paraminfo.type == F0R_PARAM_DOUBLE ){
113 mlt_properties_set ( pnum , "type" , "float" );
114 mlt_properties_set ( pnum , "minimum" , "0" );
115 mlt_properties_set ( pnum , "maximum" , "1" );
116 mlt_properties_set ( pnum , "readonly" , "no" );
117 mlt_properties_set ( pnum , "widget" , "spinner" );
118 }else
119 if ( paraminfo.type == F0R_PARAM_BOOL ){
120 mlt_properties_set ( pnum , "type" , "boolean" );
121 mlt_properties_set ( pnum , "minimum" , "0" );
122 mlt_properties_set ( pnum , "maximum" , "1" );
123 mlt_properties_set ( pnum , "readonly" , "no" );
124 }else
125 if ( paraminfo.type == F0R_PARAM_COLOR ){
126 mlt_properties_set ( pnum , "type" , "color" );
127 mlt_properties_set ( pnum , "readonly" , "no" );
128 }else
129 if ( paraminfo.type == F0R_PARAM_STRING ){
130 mlt_properties_set ( pnum , "type" , "string" );
131 mlt_properties_set ( pnum , "readonly" , "no" );
132 }
133 }
134 dlclose(handle);
135 free(name);
136
137 return metadata;
138 }
139
140 static void * load_lib( mlt_profile profile, mlt_service_type type , void* handle){
141
142 int i=0;
143 void (*f0r_get_plugin_info)(f0r_plugin_info_t*),
144 *f0r_construct , *f0r_update , *f0r_destruct,
145 (*f0r_get_param_info)(f0r_param_info_t* info, int param_index),
146 (*f0r_init)(void) , *f0r_deinit ,
147 (*f0r_set_param_value)(f0r_instance_t instance, f0r_param_t param, int param_index),
148 (*f0r_get_param_value)(f0r_instance_t instance, f0r_param_t param, int param_index),
149 (*f0r_update2) (f0r_instance_t instance, double time, const uint32_t* inframe1,
150 const uint32_t* inframe2,const uint32_t* inframe3, uint32_t* outframe);
151
152 if ( ( f0r_construct = dlsym(handle, "f0r_construct") ) &&
153 (f0r_update = dlsym(handle,"f0r_update") ) &&
154 (f0r_destruct = dlsym(handle,"f0r_destruct") ) &&
155 (f0r_get_plugin_info = dlsym(handle,"f0r_get_plugin_info") ) &&
156 (f0r_get_param_info = dlsym(handle,"f0r_get_param_info") ) &&
157 (f0r_set_param_value= dlsym(handle,"f0r_set_param_value" ) ) &&
158 (f0r_get_param_value= dlsym(handle,"f0r_get_param_value" ) ) &&
159 (f0r_init= dlsym(handle,"f0r_init" ) ) &&
160 (f0r_deinit= dlsym(handle,"f0r_deinit" ) )
161 ){
162
163 f0r_update2=dlsym(handle,"f0r_update2");
164
165 f0r_plugin_info_t info;
166 f0r_get_plugin_info(&info);
167
168 void* ret=NULL;
169 mlt_properties properties=NULL;
170
171 if (type == filter_type && info.plugin_type == F0R_PLUGIN_TYPE_FILTER ){
172 mlt_filter this = mlt_filter_new( );
173 if ( this != NULL )
174 {
175 this->process = filter_process;
176 this->close = filter_close;
177 f0r_init();
178 properties=MLT_FILTER_PROPERTIES ( this );
179
180 for (i=0;i<info.num_params;i++){
181 f0r_param_info_t pinfo;
182 f0r_get_param_info(&pinfo,i);
183
184 }
185
186 ret=this;
187 }
188 }else if (type == transition_type && info.plugin_type == F0R_PLUGIN_TYPE_MIXER2){
189 mlt_transition transition = mlt_transition_new( );
190 if ( transition != NULL )
191 {
192 transition->process = transition_process;
193 transition->close = transition_close;
194 properties=MLT_TRANSITION_PROPERTIES( transition );
195 mlt_properties_set_int(properties, "_transition_type", 1 );
196
197 ret=transition;
198 }
199 }
200 mlt_properties_set_data(properties, "_dlclose_handle", handle , sizeof (void*) , NULL , NULL );
201 mlt_properties_set_data(properties, "_dlclose", dlclose , sizeof (void*) , NULL , NULL );
202 mlt_properties_set_data(properties, "f0r_construct", f0r_construct , sizeof(void*),NULL,NULL);
203 mlt_properties_set_data(properties, "f0r_update", f0r_update , sizeof(void*),NULL,NULL);
204 if (f0r_update2)
205 mlt_properties_set_data(properties, "f0r_update2", f0r_update2 , sizeof(void*),NULL,NULL);
206 mlt_properties_set_data(properties, "f0r_destruct", f0r_destruct , sizeof(void*),NULL,NULL);
207 mlt_properties_set_data(properties, "f0r_get_plugin_info", f0r_get_plugin_info , sizeof(void*),NULL,NULL);
208 mlt_properties_set_data(properties, "f0r_get_param_info", f0r_get_param_info , sizeof(void*),NULL,NULL);
209 mlt_properties_set_data(properties, "f0r_set_param_value", f0r_set_param_value , sizeof(void*),NULL,NULL);
210 mlt_properties_set_data(properties, "f0r_get_param_value", f0r_get_param_value , sizeof(void*),NULL,NULL);
211
212
213 return ret;
214 }else{
215 printf("some was wrong\n");
216 dlerror();
217 }
218 return NULL;
219 }
220
221 static void * create_frei0r_item ( mlt_profile profile, mlt_service_type type, const char *id, void *arg){
222
223 mlt_tokeniser tokeniser = mlt_tokeniser_init ( );
224 int dircount=mlt_tokeniser_parse_new (
225 tokeniser,
226 getenv("MLT_FREI0R_PLUGIN_PATH") ? getenv("MLT_FREI0R_PLUGIN_PATH") : FREI0R_PLUGIN_PATH,
227 ":"
228 );
229 void* ret=NULL;
230 while (dircount--){
231 char soname[1024]="";
232
233 char *save_firstptr = NULL;
234 char *firstname=strtok_r(strdup(id),".",&save_firstptr);
235
236 firstname=strtok_r(NULL,".",&save_firstptr);
237 sprintf(soname,"%s/%s.so", mlt_tokeniser_get_string( tokeniser , dircount ) , firstname );
238
239 if (firstname){
240
241 void* handle=dlopen(soname,RTLD_LAZY);
242
243 if (handle ){
244 ret=load_lib ( profile , type , handle );
245 }else{
246 dlerror();
247 }
248 }
249 }
250 mlt_tokeniser_close ( tokeniser );
251 return ret;
252 }
253
254
255 MLT_REPOSITORY
256 {
257 int i=0;
258 mlt_tokeniser tokeniser = mlt_tokeniser_init ( );
259 int dircount=mlt_tokeniser_parse_new (
260 tokeniser ,
261 getenv("MLT_FREI0R_PLUGIN_PATH") ? getenv("MLT_FREI0R_PLUGIN_PATH") : FREI0R_PLUGIN_PATH,
262 ":"
263 );
264
265 while (dircount--){
266
267 mlt_properties direntries = mlt_properties_new();
268 char* dirname = mlt_tokeniser_get_string ( tokeniser , dircount ) ;
269 mlt_properties_dir_list(direntries, dirname ,"*.so",1);
270
271 for (i=0;i<mlt_properties_count(direntries);i++){
272 char* name=mlt_properties_get_value(direntries,i);
273 char* shortname=name+strlen(dirname)+1;
274 char fname[1024]="";
275
276 strcat(fname,dirname);
277 strcat(fname,shortname);
278
279 char *save_firstptr = NULL;
280 char pluginname[1024]="frei0r.";
281 char* firstname = strtok_r ( shortname , "." , &save_firstptr );
282 strcat(pluginname,firstname);
283
284 void* handle=dlopen(strcat(name,".so"),RTLD_LAZY);
285 if (handle){
286 void (*plginfo)(f0r_plugin_info_t*)=dlsym(handle,"f0r_get_plugin_info");
287
288 if (plginfo){
289 f0r_plugin_info_t info;
290 plginfo(&info);
291
292 if (firstname && info.plugin_type==F0R_PLUGIN_TYPE_FILTER){
293 MLT_REGISTER( filter_type, pluginname, create_frei0r_item );
294 MLT_REGISTER_METADATA( filter_type, pluginname, fill_param_info, strdup(name) );
295 }
296 else if (firstname && info.plugin_type==F0R_PLUGIN_TYPE_MIXER2 ){
297 MLT_REGISTER( transition_type, pluginname, create_frei0r_item );
298 MLT_REGISTER_METADATA( transition_type, pluginname, fill_param_info, strdup(name) );
299 }
300 }
301 dlclose(handle);
302 }
303 }
304 mlt_properties_close(direntries);
305 }
306 mlt_tokeniser_close ( tokeniser );
307 }