1545
|
1 #include "cmediasample.h"
|
2069
|
2 #include "wine/winerror.h"
|
168
|
3 #include <stdio.h>
|
|
4 #include <string.h>
|
7386
|
5 #include <stdlib.h>
|
1545
|
6
|
3056
|
7 static long STDCALL CMediaSample_QueryInterface(IUnknown* This,
|
7386
|
8 /* [in] */ const GUID* iid,
|
1545
|
9 /* [iid_is][out] */ void **ppv)
|
168
|
10 {
|
3056
|
11 Debug printf("CMediaSample_QueryInterface(%p) called\n", This);
|
1545
|
12 if (!ppv)
|
|
13 return E_INVALIDARG;
|
3056
|
14 if (memcmp(iid, &IID_IUnknown, sizeof(*iid)) == 0)
|
168
|
15 {
|
3056
|
16 *ppv = (void*)This;
|
|
17 ((IMediaSample*) This)->vt->AddRef(This);
|
168
|
18 return 0;
|
|
19 }
|
3056
|
20 if (memcmp(iid, &IID_IMediaSample, sizeof(*iid)) == 0)
|
168
|
21 {
|
3056
|
22 *ppv = (void*)This;
|
|
23 ((IMediaSample*) This)->vt->AddRef(This);
|
168
|
24 return 0;
|
|
25 }
|
1545
|
26 return E_NOINTERFACE;
|
168
|
27 }
|
|
28
|
1545
|
29 static long STDCALL CMediaSample_AddRef(IUnknown* This)
|
168
|
30 {
|
3056
|
31 Debug printf("CMediaSample_AddRef(%p) called\n", This);
|
168
|
32 ((CMediaSample*)This)->refcount++;
|
|
33 return 0;
|
|
34 }
|
1545
|
35
|
3056
|
36 void CMediaSample_Destroy(CMediaSample* This)
|
|
37 {
|
|
38
|
|
39 Debug printf("CMediaSample_Destroy(%p) called (ref:%d)\n", This, This->refcount);
|
|
40 free(This->vt);
|
|
41 free(This->own_block);
|
|
42 if (This->media_type.pbFormat)
|
|
43 CoTaskMemFree(This->media_type.pbFormat);
|
|
44 free(This);
|
|
45 }
|
|
46
|
1545
|
47 static long STDCALL CMediaSample_Release(IUnknown* This)
|
168
|
48 {
|
3467
|
49 CMediaSample* parent = (CMediaSample*)This;
|
3056
|
50 Debug printf("CMediaSample_Release(%p) called (new ref:%d)\n",
|
1545
|
51 This, ((CMediaSample*)This)->refcount-1);
|
3467
|
52
|
|
53 if (--((CMediaSample*) This)->refcount == 0)
|
3056
|
54 {
|
1545
|
55 parent->all->vt->ReleaseBuffer((IMemAllocator*)(parent->all),
|
|
56 (IMediaSample*)This);
|
3056
|
57 }
|
168
|
58 return 0;
|
|
59 }
|
1545
|
60
|
3467
|
61 static HRESULT STDCALL CMediaSample_GetPointer(IMediaSample* This,
|
|
62 /* [out] */ BYTE** ppBuffer)
|
168
|
63 {
|
3467
|
64 Debug printf("CMediaSample_GetPointer(%p) called -> %p, size: %d %d\n", This, ((CMediaSample*) This)->block, ((CMediaSample*)This)->actual_size, ((CMediaSample*)This)->size);
|
1545
|
65 if (!ppBuffer)
|
|
66 return E_INVALIDARG;
|
3467
|
67 *ppBuffer = (BYTE*) ((CMediaSample*) This)->block;
|
168
|
68 return 0;
|
|
69 }
|
|
70
|
1545
|
71 static long STDCALL CMediaSample_GetSize(IMediaSample * This)
|
168
|
72 {
|
3467
|
73 Debug printf("CMediaSample_GetSize(%p) called -> %d\n", This, ((CMediaSample*) This)->size);
|
|
74 return ((CMediaSample*) This)->size;
|
168
|
75 }
|
|
76
|
1545
|
77 static HRESULT STDCALL CMediaSample_GetTime(IMediaSample * This,
|
|
78 /* [out] */ REFERENCE_TIME *pTimeStart,
|
|
79 /* [out] */ REFERENCE_TIME *pTimeEnd)
|
168
|
80 {
|
3467
|
81 Debug printf("CMediaSample_GetTime(%p) called (UNIMPLIMENTED)\n", This);
|
168
|
82 return E_NOTIMPL;
|
|
83 }
|
|
84
|
1545
|
85 static HRESULT STDCALL CMediaSample_SetTime(IMediaSample * This,
|
|
86 /* [in] */ REFERENCE_TIME *pTimeStart,
|
|
87 /* [in] */ REFERENCE_TIME *pTimeEnd)
|
168
|
88 {
|
3467
|
89 Debug printf("CMediaSample_SetTime(%p) called (UNIMPLIMENTED)\n", This);
|
168
|
90 return E_NOTIMPL;
|
|
91 }
|
|
92
|
1545
|
93 static HRESULT STDCALL CMediaSample_IsSyncPoint(IMediaSample * This)
|
168
|
94 {
|
3056
|
95 Debug printf("CMediaSample_IsSyncPoint(%p) called\n", This);
|
1545
|
96 if (((CMediaSample*)This)->isSyncPoint)
|
|
97 return 0;
|
168
|
98 return 1;
|
|
99 }
|
|
100
|
1545
|
101 static HRESULT STDCALL CMediaSample_SetSyncPoint(IMediaSample * This,
|
|
102 long bIsSyncPoint)
|
168
|
103 {
|
3056
|
104 Debug printf("CMediaSample_SetSyncPoint(%p) called\n", This);
|
3467
|
105 ((CMediaSample*)This)->isSyncPoint = bIsSyncPoint;
|
168
|
106 return 0;
|
|
107 }
|
|
108
|
1545
|
109 static HRESULT STDCALL CMediaSample_IsPreroll(IMediaSample * This)
|
168
|
110 {
|
3056
|
111 Debug printf("CMediaSample_IsPreroll(%p) called\n", This);
|
1545
|
112
|
|
113 if (((CMediaSample*)This)->isPreroll)
|
168
|
114 return 0;//S_OK
|
1545
|
115
|
|
116 return 1;//S_FALSE
|
168
|
117 }
|
|
118
|
1545
|
119 static HRESULT STDCALL CMediaSample_SetPreroll(IMediaSample * This,
|
|
120 long bIsPreroll)
|
168
|
121 {
|
3056
|
122 Debug printf("CMediaSample_SetPreroll(%p) called\n", This);
|
168
|
123 ((CMediaSample*)This)->isPreroll=bIsPreroll;
|
|
124 return 0;
|
|
125 }
|
|
126
|
3056
|
127 static long STDCALL CMediaSample_GetActualDataLength(IMediaSample* This)
|
168
|
128 {
|
3056
|
129 Debug printf("CMediaSample_GetActualDataLength(%p) called -> %d\n", This, ((CMediaSample*)This)->actual_size);
|
168
|
130 return ((CMediaSample*)This)->actual_size;
|
|
131 }
|
|
132
|
3056
|
133 static HRESULT STDCALL CMediaSample_SetActualDataLength(IMediaSample* This,
|
1545
|
134 long __MIDL_0010)
|
168
|
135 {
|
3467
|
136 CMediaSample* cms = (CMediaSample*)This;
|
3056
|
137 Debug printf("CMediaSample_SetActualDataLength(%p, %ld) called\n", This, __MIDL_0010);
|
3467
|
138 if (__MIDL_0010 > cms->size)
|
168
|
139 {
|
3467
|
140 char* c = cms->own_block;
|
|
141 Debug printf(" CMediaSample - buffer overflow %ld %d %p %p\n",
|
|
142 __MIDL_0010, ((CMediaSample*)This)->size, cms->own_block, cms->block);
|
|
143 cms->own_block = realloc(cms->own_block, __MIDL_0010);
|
|
144 if (c == cms->block)
|
|
145 cms->block = cms->own_block;
|
|
146 cms->size = __MIDL_0010;
|
168
|
147 }
|
3467
|
148 cms->actual_size = __MIDL_0010;
|
168
|
149 return 0;
|
|
150 }
|
|
151
|
3056
|
152 static HRESULT STDCALL CMediaSample_GetMediaType(IMediaSample* This,
|
|
153 AM_MEDIA_TYPE** ppMediaType)
|
168
|
154 {
|
3056
|
155 AM_MEDIA_TYPE* t;
|
|
156 Debug printf("CMediaSample_GetMediaType(%p) called\n", This);
|
168
|
157 if(!ppMediaType)
|
1545
|
158 return E_INVALIDARG;
|
168
|
159 if(!((CMediaSample*)This)->type_valid)
|
|
160 {
|
|
161 *ppMediaType=0;
|
|
162 return 1;
|
|
163 }
|
3056
|
164
|
|
165 t = &((CMediaSample*)This)->media_type;
|
|
166 // if(t.pbFormat)CoTaskMemFree(t.pbFormat);
|
|
167 (*ppMediaType) = (AM_MEDIA_TYPE*)CoTaskMemAlloc(sizeof(AM_MEDIA_TYPE));
|
|
168 **ppMediaType = *t;
|
|
169 (*ppMediaType)->pbFormat = (char*)CoTaskMemAlloc(t->cbFormat);
|
|
170 memcpy((*ppMediaType)->pbFormat, t->pbFormat, t->cbFormat);
|
|
171 // *ppMediaType=0; //media type was not changed
|
1545
|
172 return 0;
|
168
|
173 }
|
|
174
|
1545
|
175 static HRESULT STDCALL CMediaSample_SetMediaType(IMediaSample * This,
|
|
176 AM_MEDIA_TYPE *pMediaType)
|
168
|
177 {
|
3056
|
178 AM_MEDIA_TYPE* t;
|
|
179 Debug printf("CMediaSample_SetMediaType(%p) called\n", This);
|
1545
|
180 if (!pMediaType)
|
|
181 return E_INVALIDARG;
|
3056
|
182 t = &((CMediaSample*)This)->media_type;
|
|
183 if (t->pbFormat)
|
|
184 CoTaskMemFree(t->pbFormat);
|
|
185 t = pMediaType;
|
3467
|
186 if (t->cbFormat)
|
|
187 {
|
|
188 t->pbFormat = (char*)CoTaskMemAlloc(t->cbFormat);
|
|
189 memcpy(t->pbFormat, pMediaType->pbFormat, t->cbFormat);
|
|
190 }
|
|
191 else
|
|
192 t->pbFormat = 0;
|
|
193 ((CMediaSample*) This)->type_valid=1;
|
1545
|
194
|
168
|
195 return 0;
|
|
196 }
|
|
197
|
1545
|
198 static HRESULT STDCALL CMediaSample_IsDiscontinuity(IMediaSample * This)
|
168
|
199 {
|
3056
|
200 Debug printf("CMediaSample_IsDiscontinuity(%p) called\n", This);
|
3467
|
201 return ((CMediaSample*) This)->isDiscontinuity;
|
168
|
202 }
|
|
203
|
1545
|
204 static HRESULT STDCALL CMediaSample_SetDiscontinuity(IMediaSample * This,
|
|
205 long bDiscontinuity)
|
168
|
206 {
|
3467
|
207 Debug printf("CMediaSample_SetDiscontinuity(%p) called (%ld)\n", This, bDiscontinuity);
|
|
208 ((CMediaSample*) This)->isDiscontinuity = bDiscontinuity;
|
|
209 return 0;
|
1545
|
210 }
|
|
211
|
|
212 static HRESULT STDCALL CMediaSample_GetMediaTime(IMediaSample * This,
|
|
213 /* [out] */ LONGLONG *pTimeStart,
|
|
214 /* [out] */ LONGLONG *pTimeEnd)
|
|
215 {
|
3056
|
216 Debug printf("CMediaSample_GetMediaTime(%p) called\n", This);
|
3467
|
217 if (pTimeStart)
|
|
218 *pTimeStart = ((CMediaSample*) This)->time_start;
|
|
219 if (pTimeEnd)
|
|
220 *pTimeEnd = ((CMediaSample*) This)->time_end;
|
|
221 return 0;
|
1545
|
222 }
|
|
223
|
|
224 static HRESULT STDCALL CMediaSample_SetMediaTime(IMediaSample * This,
|
|
225 /* [in] */ LONGLONG *pTimeStart,
|
|
226 /* [in] */ LONGLONG *pTimeEnd)
|
|
227 {
|
3056
|
228 Debug printf("CMediaSample_SetMediaTime(%p) called\n", This);
|
3467
|
229 if (pTimeStart)
|
|
230 ((CMediaSample*) This)->time_start = *pTimeStart;
|
|
231 if (pTimeEnd)
|
|
232 ((CMediaSample*) This)->time_end = *pTimeEnd;
|
|
233 return 0;
|
168
|
234 }
|
|
235
|
3467
|
236 // extension for direct memory write or decompressed data
|
3056
|
237 static void CMediaSample_SetPointer(CMediaSample* This, char* pointer)
|
168
|
238 {
|
3056
|
239 Debug printf("CMediaSample_SetPointer(%p) called -> %p\n", This, pointer);
|
|
240 if (pointer)
|
|
241 This->block = pointer;
|
|
242 else
|
|
243 This->block = This->own_block;
|
|
244 }
|
1545
|
245
|
3056
|
246 static void CMediaSample_ResetPointer(CMediaSample* This)
|
|
247 {
|
|
248 Debug printf("CMediaSample_ResetPointer(%p) called\n", This);
|
|
249 This->block = This->own_block;
|
168
|
250 }
|
|
251
|
3056
|
252 CMediaSample* CMediaSampleCreate(IMemAllocator* allocator, int _size)
|
168
|
253 {
|
3467
|
254 CMediaSample* This = (CMediaSample*) malloc(sizeof(CMediaSample));
|
|
255 if (!This)
|
|
256 return NULL;
|
|
257
|
|
258 // some hack here!
|
|
259 // it looks like Acelp decoder is actually accessing
|
|
260 // the allocated memory before it sets the new size for it ???
|
|
261 // -- maybe it's being initialized with wrong parameters
|
|
262 // anyway this is fixes the problem somehow with some reserves
|
|
263 //
|
|
264 // using different trick for now - in DS_Audio modify sample size
|
|
265 //if (_size < 0x1000)
|
|
266 // _size = (_size + 0xfff) & ~0xfff;
|
|
267
|
3056
|
268 This->vt = (IMediaSample_vt*) malloc(sizeof(IMediaSample_vt));
|
3467
|
269 This->own_block = (char*) malloc(_size);
|
|
270 This->media_type.pbFormat = 0;
|
|
271
|
|
272 if (!This->vt || !This->own_block)
|
|
273 {
|
|
274 CMediaSample_Destroy(This);
|
|
275 return NULL;
|
|
276 }
|
3056
|
277
|
|
278 This->vt->QueryInterface = CMediaSample_QueryInterface;
|
|
279 This->vt->AddRef = CMediaSample_AddRef;
|
|
280 This->vt->Release = CMediaSample_Release;
|
|
281 This->vt->GetPointer = CMediaSample_GetPointer;
|
|
282 This->vt->GetSize = CMediaSample_GetSize;
|
|
283 This->vt->GetTime = CMediaSample_GetTime;
|
|
284 This->vt->SetTime = CMediaSample_SetTime;
|
|
285 This->vt->IsSyncPoint = CMediaSample_IsSyncPoint;
|
|
286 This->vt->SetSyncPoint = CMediaSample_SetSyncPoint;
|
|
287 This->vt->IsPreroll = CMediaSample_IsPreroll;
|
|
288 This->vt->SetPreroll = CMediaSample_SetPreroll;
|
|
289 This->vt->GetActualDataLength = CMediaSample_GetActualDataLength;
|
|
290 This->vt->SetActualDataLength = CMediaSample_SetActualDataLength;
|
|
291 This->vt->GetMediaType = CMediaSample_GetMediaType;
|
|
292 This->vt->SetMediaType = CMediaSample_SetMediaType;
|
|
293 This->vt->IsDiscontinuity = CMediaSample_IsDiscontinuity;
|
|
294 This->vt->SetDiscontinuity = CMediaSample_SetDiscontinuity;
|
|
295 This->vt->GetMediaTime = CMediaSample_GetMediaTime;
|
|
296 This->vt->SetMediaTime = CMediaSample_SetMediaTime;
|
|
297
|
|
298 This->all = allocator;
|
|
299 This->size = _size;
|
|
300 This->refcount = 0; // increased by MemAllocator
|
|
301 This->actual_size = 0;
|
|
302 This->isPreroll = 0;
|
3467
|
303 This->isDiscontinuity = 1;
|
|
304 This->time_start = 0;
|
|
305 This->time_end = 0;
|
3056
|
306 This->type_valid = 0;
|
|
307 This->block = This->own_block;
|
|
308
|
|
309 This->SetPointer = CMediaSample_SetPointer;
|
|
310 This->ResetPointer = CMediaSample_ResetPointer;
|
|
311
|
|
312 Debug printf("CMediaSample_Create(%p) called - sample size %d, buffer %p\n",
|
|
313 This, This->size, This->block);
|
|
314
|
|
315 return This;
|
168
|
316 }
|