]> git.sesse.net Git - mlt/blob - src/modules/core/producer_loader.c
4092bd1efca2aa517294f10e81657e0e8ade7047
[mlt] / src / modules / core / producer_loader.c
1 /*
2  * producer_loader.c -- auto-load producer by file name extension
3  * Copyright (C) 2003-2009 Ushodaya Enterprises Limited
4  * Author: Charles Yates <charles.yates@pandora.be>
5  *
6  * This library is free software; you can redistribute it and/or
7  * modify it under the terms of the GNU Lesser General Public
8  * License as published by the Free Software Foundation; either
9  * version 2.1 of the License, or (at your option) any later version.
10  *
11  * This library is distributed in the hope that it will be useful,
12  * but WITHOUT ANY WARRANTY; without even the implied warranty of
13  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
14  * Lesser General Public License for more details.
15  *
16  * You should have received a copy of the GNU Lesser General Public
17  * License along with this library; if not, write to the Free Software
18  * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
19  */
20
21 #include <stdio.h>
22 #include <stdlib.h>
23 #include <string.h>
24 #include <ctype.h>
25 #include <fnmatch.h>
26 #include <assert.h>
27
28 #include <framework/mlt.h>
29
30 static mlt_properties dictionary = NULL;
31 static mlt_properties normalisers = NULL;
32
33 static mlt_producer create_from( mlt_profile profile, char *file, char *services )
34 {
35         mlt_producer producer = NULL;
36         char *temp = strdup( services );
37         char *service = temp;
38         do
39         {
40                 char *p = strchr( service, ',' );
41                 if ( p != NULL )
42                         *p ++ = '\0';
43                 producer = mlt_factory_producer( profile, service, file );
44                 service = p;
45         }
46         while ( producer == NULL && service != NULL );
47         free( temp );
48         return producer;
49 }
50
51 static mlt_producer create_producer( mlt_profile profile, char *file )
52 {
53         mlt_producer result = NULL;
54
55         // 1st Line - check for service:resource handling
56         if ( strchr( file, ':' ) )
57         {
58                 char *temp = strdup( file );
59                 char *service = temp;
60                 char *resource = strchr( temp, ':' );
61                 *resource ++ = '\0';
62                 result = mlt_factory_producer( profile, service, resource );
63                 free( temp );
64         }
65
66         // 2nd Line preferences
67         if ( result == NULL )
68         {
69                 int i = 0;
70                 char *lookup = strdup( file );
71                 char *p = lookup;
72
73                 // We only need to load the dictionary once
74                 if ( dictionary == NULL )
75                 {
76                         char temp[ 1024 ];
77                         sprintf( temp, "%s/core/loader.dict", mlt_environment( "MLT_DATA" ) );
78                         dictionary = mlt_properties_load( temp );
79                         mlt_factory_register_for_clean_up( dictionary, ( mlt_destructor )mlt_properties_close );
80                 }
81
82                 // Convert the lookup string to lower case
83                 while ( *p )
84                 {
85                         *p = tolower( *p );
86                         p ++;
87                 }
88
89                 // Iterate through the dictionary
90                 for ( i = 0; result == NULL && i < mlt_properties_count( dictionary ); i ++ )
91                 {
92                         char *name = mlt_properties_get_name( dictionary, i );
93                         if ( fnmatch( name, lookup, 0 ) == 0 )
94                                 result = create_from( profile, file, mlt_properties_get_value( dictionary, i ) );
95                 }
96
97                 free( lookup );
98         }
99
100         // Finally, try just loading as service
101         if ( result == NULL )
102                 result = mlt_factory_producer( profile, file, NULL );
103
104         return result;
105 }
106
107 static void create_filter( mlt_profile profile, mlt_producer producer, char *effect, int *created )
108 {
109         char *id = strdup( effect );
110         char *arg = strchr( id, ':' );
111         if ( arg != NULL )
112                 *arg ++ = '\0';
113
114         // The swscale and avcolor_space filters require resolution as arg to test compatibility
115         if ( strncmp( effect, "swscale", 7 ) == 0 || strncmp( effect, "avcolo", 6 ) == 0 )
116                 arg = (char*) mlt_properties_get_int( MLT_PRODUCER_PROPERTIES( producer ), "_real_width" );
117
118         mlt_filter filter = mlt_factory_filter( profile, id, arg );
119         if ( filter != NULL )
120         {
121                 mlt_properties_set_int( MLT_FILTER_PROPERTIES( filter ), "_loader", 1 );
122                 mlt_producer_attach( producer, filter );
123                 mlt_filter_close( filter );
124                 *created = 1;
125         }
126         free( id );
127 }
128
129 static void attach_normalisers( mlt_profile profile, mlt_producer producer )
130 {
131         // Loop variable
132         int i;
133
134         // Tokeniser
135         mlt_tokeniser tokeniser = mlt_tokeniser_init( );
136
137         // We only need to load the normalising properties once
138         if ( normalisers == NULL )
139         {
140                 char temp[ 1024 ];
141                 sprintf( temp, "%s/core/loader.ini", mlt_environment( "MLT_DATA" ) );
142                 normalisers = mlt_properties_load( temp );
143                 mlt_factory_register_for_clean_up( normalisers, ( mlt_destructor )mlt_properties_close );
144         }
145
146         // Apply normalisers
147         for ( i = 0; i < mlt_properties_count( normalisers ); i ++ )
148         {
149                 int j = 0;
150                 int created = 0;
151                 char *value = mlt_properties_get_value( normalisers, i );
152                 mlt_tokeniser_parse_new( tokeniser, value, "," );
153                 for ( j = 0; !created && j < mlt_tokeniser_count( tokeniser ); j ++ )
154                         create_filter( profile, producer, mlt_tokeniser_get_string( tokeniser, j ), &created );
155         }
156
157         // Close the tokeniser
158         mlt_tokeniser_close( tokeniser );
159 }
160
161 mlt_producer producer_loader_init( mlt_profile profile, mlt_service_type type, const char *id, char *arg )
162 {
163         // Create the producer 
164         mlt_producer producer = NULL;
165         mlt_properties properties = NULL;
166
167         if ( arg != NULL )
168                 producer = create_producer( profile, arg );
169
170         if ( producer != NULL )
171                 properties = MLT_PRODUCER_PROPERTIES( producer );
172
173         // Attach filters if we have a producer and it isn't already xml'd :-)
174         if ( producer && strcmp( id, "abnormal" ) &&
175                 mlt_properties_get( properties, "xml" ) == NULL &&
176                 mlt_properties_get( properties, "_xml" ) == NULL &&
177                 mlt_properties_get( properties, "loader_normalised" ) == NULL )
178                 attach_normalisers( profile, producer );
179         
180         if ( producer )
181         {
182                 // Always let the image and audio be converted
183                 int created = 0;
184                 create_filter( profile, producer, "avcolor_space", &created );
185                 if ( !created )
186                         create_filter( profile, producer, "imageconvert", &created );
187                 create_filter( profile, producer, "audioconvert", &created );
188         }
189
190         // Now make sure we don't lose our identity
191         if ( properties != NULL )
192                 mlt_properties_set_int( properties, "_mlt_service_hidden", 1 );
193
194         // Return the producer
195         return producer;
196 }