~ [ source navigation ] ~ [ diff markup ] ~ [ identifier search ] ~ [ freetext search ] ~ [ file search ] ~

Wine Cross Reference
wine/dlls/inetcomm/mimeintl.c

Version: ~ [ wine-1.1.33 ] ~ [ wine-1.1.32 ] ~ [ wine-1.1.31 ] ~ [ wine-1.1.30 ] ~ [ wine-1.1.29 ] ~ [ wine-1.1.28 ] ~ [ wine-1.1.27 ] ~ [ wine-1.1.26 ] ~ [ wine-1.1.25 ] ~ [ wine-1.1.24 ] ~ [ wine-1.1.23 ] ~ [ wine-1.1.22 ] ~ [ wine-1.1.21 ] ~ [ wine-1.1.20 ] ~ [ wine-1.1.19 ] ~ [ wine-1.1.18 ] ~ [ wine-1.1.17 ] ~ [ wine-1.1.16 ] ~ [ wine-1.1.15 ] ~ [ wine-1.1.14 ] ~ [ wine-1.1.13 ] ~ [ wine-1.1.12 ] ~ [ wine-1.1.11 ] ~ [ wine-1.1.10 ] ~ [ wine-1.1.9 ] ~ [ wine-1.1.8 ] ~ [ wine-1.1.7 ] ~ [ wine-1.0.1 ] ~ [ wine-1.1.6 ] ~ [ wine-1.1.5 ] ~ [ wine-1.1.4 ] ~ [ wine-1.1.3 ] ~ [ wine-1.1.2 ] ~ [ wine-1.1.1 ] ~ [ wine-1.1.0 ] ~ [ wine-1.0 ] ~

  1 /*
  2  * MIME OLE International interface
  3  *
  4  * Copyright 2008 Huw Davies for CodeWeavers
  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., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
 19  */
 20 
 21 #define COBJMACROS
 22 #define NONAMELESSUNION
 23 
 24 #include <stdarg.h>
 25 #include <stdio.h>
 26 
 27 #include "windef.h"
 28 #include "winbase.h"
 29 #include "winuser.h"
 30 #include "winnls.h"
 31 #include "objbase.h"
 32 #include "ole2.h"
 33 #include "mimeole.h"
 34 #include "mlang.h"
 35 
 36 #include "wine/list.h"
 37 #include "wine/unicode.h"
 38 #include "wine/debug.h"
 39 
 40 #include "inetcomm_private.h"
 41 
 42 WINE_DEFAULT_DEBUG_CHANNEL(inetcomm);
 43 
 44 typedef struct
 45 {
 46     struct list entry;
 47     INETCSETINFO cs_info;
 48 } charset_entry;
 49 
 50 typedef struct
 51 {
 52     const IMimeInternationalVtbl *lpVtbl;
 53     LONG refs;
 54     CRITICAL_SECTION cs;
 55 
 56     struct list charsets;
 57     LONG next_charset_handle;
 58     HCHARSET default_charset;
 59 } internat;
 60 
 61 static inline internat *impl_from_IMimeInternational( IMimeInternational *iface )
 62 {
 63     return (internat *)((char*)iface - FIELD_OFFSET(internat, lpVtbl));
 64 }
 65 
 66 static inline HRESULT get_mlang(IMultiLanguage **ml)
 67 {
 68     return CoCreateInstance(&CLSID_CMultiLanguage, NULL,  CLSCTX_INPROC_SERVER | CLSCTX_INPROC_HANDLER,
 69                             &IID_IMultiLanguage, (void **)ml);
 70 }
 71 
 72 static HRESULT WINAPI MimeInternat_QueryInterface( IMimeInternational *iface, REFIID riid, LPVOID *ppobj )
 73 {
 74     if (IsEqualGUID(riid, &IID_IUnknown) ||
 75         IsEqualGUID(riid, &IID_IMimeInternational))
 76     {
 77         IMimeInternational_AddRef( iface );
 78         *ppobj = iface;
 79         return S_OK;
 80     }
 81 
 82     FIXME("interface %s not implemented\n", debugstr_guid(riid));
 83     return E_NOINTERFACE;
 84 }
 85 
 86 static ULONG WINAPI MimeInternat_AddRef( IMimeInternational *iface )
 87 {
 88     internat *This = impl_from_IMimeInternational( iface );
 89     return InterlockedIncrement(&This->refs);
 90 }
 91 
 92 static ULONG WINAPI MimeInternat_Release( IMimeInternational *iface )
 93 {
 94     internat *This = impl_from_IMimeInternational( iface );
 95     ULONG refs;
 96 
 97     refs = InterlockedDecrement(&This->refs);
 98     if (!refs)
 99     {
100         charset_entry *charset, *cursor2;
101 
102         LIST_FOR_EACH_ENTRY_SAFE(charset, cursor2, &This->charsets, charset_entry, entry)
103         {
104             list_remove(&charset->entry);
105             HeapFree(GetProcessHeap(), 0, charset);
106         }
107         HeapFree(GetProcessHeap(), 0, This);
108     }
109 
110     return refs;
111 }
112 
113 static HRESULT WINAPI MimeInternat_SetDefaultCharset(IMimeInternational *iface, HCHARSET hCharset)
114 {
115     internat *This = impl_from_IMimeInternational( iface );
116 
117     TRACE("(%p)->(%p)\n", iface, hCharset);
118 
119     if(hCharset == NULL) return E_INVALIDARG;
120     /* FIXME check hCharset is valid */
121 
122     InterlockedExchangePointer(&This->default_charset, hCharset);
123 
124     return S_OK;
125 }
126 
127 static HRESULT WINAPI MimeInternat_GetDefaultCharset(IMimeInternational *iface, LPHCHARSET phCharset)
128 {
129     internat *This = impl_from_IMimeInternational( iface );
130     HRESULT hr = S_OK;
131 
132     TRACE("(%p)->(%p)\n", iface, phCharset);
133 
134     if(This->default_charset == NULL)
135     {
136         HCHARSET hcs;
137         hr = IMimeInternational_GetCodePageCharset(iface, GetACP(), CHARSET_BODY, &hcs);
138         if(SUCCEEDED(hr))
139             InterlockedCompareExchangePointer(&This->default_charset, hcs, NULL);
140     }
141     *phCharset = This->default_charset;
142 
143     return hr;
144 }
145 
146 static HRESULT mlang_getcodepageinfo(UINT cp, MIMECPINFO *mlang_cp_info)
147 {
148     HRESULT hr;
149     IMultiLanguage *ml;
150 
151     hr = get_mlang(&ml);
152 
153     if(SUCCEEDED(hr))
154     {
155         hr = IMultiLanguage_GetCodePageInfo(ml, cp, mlang_cp_info);
156         IMultiLanguage_Release(ml);
157     }
158     return hr;
159 }
160 
161 static HRESULT WINAPI MimeInternat_GetCodePageCharset(IMimeInternational *iface, CODEPAGEID cpiCodePage,
162                                                       CHARSETTYPE ctCsetType,
163                                                       LPHCHARSET phCharset)
164 {
165     HRESULT hr;
166     MIMECPINFO mlang_cp_info;
167 
168     TRACE("(%p)->(%d, %d, %p)\n", iface, cpiCodePage, ctCsetType, phCharset);
169 
170     *phCharset = NULL;
171 
172     hr = mlang_getcodepageinfo(cpiCodePage, &mlang_cp_info);
173     if(SUCCEEDED(hr))
174     {
175         const WCHAR *charset_nameW = NULL;
176         char *charset_name;
177         DWORD len;
178 
179         switch(ctCsetType)
180         {
181         case CHARSET_BODY:
182             charset_nameW = mlang_cp_info.wszBodyCharset;
183             break;
184         case CHARSET_HEADER:
185             charset_nameW = mlang_cp_info.wszHeaderCharset;
186             break;
187         case CHARSET_WEB:
188             charset_nameW = mlang_cp_info.wszWebCharset;
189             break;
190         default:
191             return MIME_E_INVALID_CHARSET_TYPE;
192         }
193         len = WideCharToMultiByte(CP_ACP, 0, charset_nameW, -1, NULL, 0, NULL, NULL);
194         charset_name = HeapAlloc(GetProcessHeap(), 0, len);
195         WideCharToMultiByte(CP_ACP, 0, charset_nameW, -1, charset_name, len, NULL, NULL);
196         hr = IMimeInternational_FindCharset(iface, charset_name, phCharset);
197         HeapFree(GetProcessHeap(), 0, charset_name);
198     }
199     return hr;
200 }
201 
202 static HRESULT mlang_getcsetinfo(const char *charset, MIMECSETINFO *mlang_info)
203 {
204     DWORD len = MultiByteToWideChar(CP_ACP, 0, charset, -1, NULL, 0);
205     BSTR bstr = SysAllocStringLen(NULL, len - 1);
206     HRESULT hr;
207     IMultiLanguage *ml;
208 
209     MultiByteToWideChar(CP_ACP, 0, charset, -1, bstr, len);
210 
211     hr = get_mlang(&ml);
212 
213     if(SUCCEEDED(hr))
214     {
215         hr = IMultiLanguage_GetCharsetInfo(ml, bstr, mlang_info);
216         IMultiLanguage_Release(ml);
217     }
218     SysFreeString(bstr);
219     if(FAILED(hr)) hr = MIME_E_NOT_FOUND;
220     return hr;
221 }
222 
223 static HCHARSET add_charset(struct list *list, MIMECSETINFO *mlang_info, HCHARSET handle)
224 {
225     charset_entry *charset = HeapAlloc(GetProcessHeap(), 0, sizeof(*charset));
226 
227     WideCharToMultiByte(CP_ACP, 0, mlang_info->wszCharset, -1,
228                         charset->cs_info.szName, sizeof(charset->cs_info.szName), NULL, NULL);
229     charset->cs_info.cpiWindows = mlang_info->uiCodePage;
230     charset->cs_info.cpiInternet = mlang_info->uiInternetEncoding;
231     charset->cs_info.hCharset = handle;
232     charset->cs_info.dwReserved1 = 0;
233     list_add_head(list, &charset->entry);
234 
235     return charset->cs_info.hCharset;
236 }
237 
238 static HRESULT WINAPI MimeInternat_FindCharset(IMimeInternational *iface, LPCSTR pszCharset,
239                                                LPHCHARSET phCharset)
240 {
241     internat *This = impl_from_IMimeInternational( iface );
242     HRESULT hr = MIME_E_NOT_FOUND;
243     charset_entry *charset;
244 
245     TRACE("(%p)->(%s, %p)\n", iface, debugstr_a(pszCharset), phCharset);
246 
247     *phCharset = NULL;
248 
249     EnterCriticalSection(&This->cs);
250 
251     LIST_FOR_EACH_ENTRY(charset, &This->charsets, charset_entry, entry)
252     {
253         if(!lstrcmpiA(charset->cs_info.szName, pszCharset))
254         {
255             *phCharset = charset->cs_info.hCharset;
256             hr = S_OK;
257             break;
258         }
259     }
260 
261     if(hr == MIME_E_NOT_FOUND)
262     {
263         MIMECSETINFO mlang_info;
264 
265         LeaveCriticalSection(&This->cs);
266         hr = mlang_getcsetinfo(pszCharset, &mlang_info);
267         EnterCriticalSection(&This->cs);
268 
269         if(SUCCEEDED(hr))
270             *phCharset = add_charset(&This->charsets, &mlang_info,
271                                      (HCHARSET)InterlockedIncrement(&This->next_charset_handle));
272     }
273 
274     LeaveCriticalSection(&This->cs);
275     return hr;
276 }
277 
278 static HRESULT WINAPI MimeInternat_GetCharsetInfo(IMimeInternational *iface, HCHARSET hCharset,
279                                                   LPINETCSETINFO pCsetInfo)
280 {
281     internat *This = impl_from_IMimeInternational( iface );
282     HRESULT hr = MIME_E_INVALID_HANDLE;
283     charset_entry *charset;
284 
285     TRACE("(%p)->(%p, %p)\n", iface, hCharset, pCsetInfo);
286 
287     EnterCriticalSection(&This->cs);
288 
289     LIST_FOR_EACH_ENTRY(charset, &This->charsets, charset_entry, entry)
290     {
291         if(charset->cs_info.hCharset ==  hCharset)
292         {
293             *pCsetInfo = charset->cs_info;
294             hr = S_OK;
295             break;
296         }
297     }
298 
299     LeaveCriticalSection(&This->cs);
300 
301     return hr;
302 }
303 
304 static HRESULT WINAPI MimeInternat_GetCodePageInfo(IMimeInternational *iface, CODEPAGEID cpiCodePage,
305                                                    LPCODEPAGEINFO pCodePageInfo)
306 {
307     FIXME("stub\n");
308     return E_NOTIMPL;
309 }
310 
311 static HRESULT WINAPI MimeInternat_CanConvertCodePages(IMimeInternational *iface, CODEPAGEID cpiSource,
312                                                        CODEPAGEID cpiDest)
313 {
314     HRESULT hr;
315     IMultiLanguage *ml;
316 
317     TRACE("(%p)->(%d, %d)\n", iface, cpiSource, cpiDest);
318 
319     /* Could call mlang.IsConvertINetStringAvailable() to avoid the COM overhead if need be. */
320 
321     hr = get_mlang(&ml);
322     if(SUCCEEDED(hr))
323     {
324         hr = IMultiLanguage_IsConvertible(ml, cpiSource, cpiDest);
325         IMultiLanguage_Release(ml);
326     }
327 
328     return hr;
329 }
330 
331 static HRESULT WINAPI MimeInternat_DecodeHeader(IMimeInternational *iface, HCHARSET hCharset,
332                                                 LPCSTR pszData,
333                                                 LPPROPVARIANT pDecoded,
334                                                 LPRFC1522INFO pRfc1522Info)
335 {
336     FIXME("stub\n");
337     return E_NOTIMPL;
338 }
339 
340 static HRESULT WINAPI MimeInternat_EncodeHeader(IMimeInternational *iface, HCHARSET hCharset,
341                                                 LPPROPVARIANT pData,
342                                                 LPSTR *ppszEncoded,
343                                                 LPRFC1522INFO pRfc1522Info)
344 {
345     FIXME("stub\n");
346     return E_NOTIMPL;
347 }
348 
349 static HRESULT WINAPI MimeInternat_ConvertBuffer(IMimeInternational *iface, CODEPAGEID cpiSource,
350                                                  CODEPAGEID cpiDest, LPBLOB pIn, LPBLOB pOut,
351                                                  ULONG *pcbRead)
352 {
353     HRESULT hr;
354     IMultiLanguage *ml;
355 
356     TRACE("(%p)->(%d, %d, %p, %p, %p)\n", iface, cpiSource, cpiDest, pIn, pOut, pcbRead);
357 
358     *pcbRead = 0;
359     pOut->cbSize = 0;
360 
361     /* Could call mlang.ConvertINetString() to avoid the COM overhead if need be. */
362 
363     hr = get_mlang(&ml);
364     if(SUCCEEDED(hr))
365     {
366         DWORD mode = 0;
367         UINT in_size = pIn->cbSize, out_size;
368 
369         hr = IMultiLanguage_ConvertString(ml, &mode, cpiSource, cpiDest, pIn->pBlobData, &in_size,
370                                           NULL, &out_size);
371         if(hr == S_OK) /* S_FALSE means the conversion could not be performed */
372         {
373             pOut->pBlobData = CoTaskMemAlloc(out_size);
374             if(!pOut->pBlobData)
375                 hr = E_OUTOFMEMORY;
376             else
377             {
378                 mode = 0;
379                 in_size = pIn->cbSize;
380                 hr = IMultiLanguage_ConvertString(ml, &mode, cpiSource, cpiDest, pIn->pBlobData, &in_size,
381                                                   pOut->pBlobData, &out_size);
382 
383                 if(hr == S_OK)
384                 {
385                     *pcbRead = in_size;
386                     pOut->cbSize = out_size;
387                 }
388                 else
389                     CoTaskMemFree(pOut->pBlobData);
390             }
391         }
392         IMultiLanguage_Release(ml);
393     }
394 
395     return hr;
396 }
397 
398 static HRESULT WINAPI MimeInternat_ConvertString(IMimeInternational *iface, CODEPAGEID cpiSource,
399                                                  CODEPAGEID cpiDest, LPPROPVARIANT pIn,
400                                                  LPPROPVARIANT pOut)
401 {
402     HRESULT hr;
403     int src_len;
404     IMultiLanguage *ml;
405 
406     TRACE("(%p)->(%d, %d, %p %p)\n", iface, cpiSource, cpiDest, pIn, pOut);
407 
408     switch(pIn->vt)
409     {
410     case VT_LPSTR:
411         if(cpiSource == CP_UNICODE) cpiSource = GetACP();
412         src_len = strlen(pIn->u.pszVal);
413         break;
414     case VT_LPWSTR:
415         cpiSource = CP_UNICODE;
416         src_len = strlenW(pIn->u.pwszVal) * sizeof(WCHAR);
417         break;
418     default:
419         return E_INVALIDARG;
420     }
421 
422     hr = get_mlang(&ml);
423     if(SUCCEEDED(hr))
424     {
425         DWORD mode = 0;
426         UINT in_size = src_len, out_size;
427 
428         hr = IMultiLanguage_ConvertString(ml, &mode, cpiSource, cpiDest, (BYTE*)pIn->u.pszVal, &in_size,
429                                           NULL, &out_size);
430         if(hr == S_OK) /* S_FALSE means the conversion could not be performed */
431         {
432             out_size += (cpiDest == CP_UNICODE) ? sizeof(WCHAR) : sizeof(char);
433 
434             pOut->u.pszVal = CoTaskMemAlloc(out_size);
435             if(!pOut->u.pszVal)
436                 hr = E_OUTOFMEMORY;
437             else
438             {
439                 mode = 0;
440                 in_size = src_len;
441                 hr = IMultiLanguage_ConvertString(ml, &mode, cpiSource, cpiDest, (BYTE*)pIn->u.pszVal, &in_size,
442                                                   (BYTE*)pOut->u.pszVal, &out_size);
443 
444                 if(hr == S_OK)
445                 {
446                     if(cpiDest == CP_UNICODE)
447                     {
448                         pOut->u.pwszVal[out_size / sizeof(WCHAR)] = 0;
449                         pOut->vt = VT_LPWSTR;
450                     }
451                     else
452                     {
453                         pOut->u.pszVal[out_size] = '\0';
454                         pOut->vt = VT_LPSTR;
455                     }
456                 }
457                 else
458                     CoTaskMemFree(pOut->u.pszVal);
459             }
460         }
461         IMultiLanguage_Release(ml);
462     }
463     return hr;
464 }
465 
466 static HRESULT WINAPI MimeInternat_MLANG_ConvertInetReset(IMimeInternational *iface)
467 {
468     FIXME("stub\n");
469     return E_NOTIMPL;
470 }
471 
472 static HRESULT WINAPI MimeInternat_MLANG_ConvertInetString(IMimeInternational *iface, CODEPAGEID cpiSource,
473                                                            CODEPAGEID cpiDest,
474                                                            LPCSTR pSource,
475                                                            int *pnSizeOfSource,
476                                                            LPSTR pDestination,
477                                                            int *pnDstSize)
478 {
479     FIXME("stub\n");
480     return E_NOTIMPL;
481 }
482 
483 static HRESULT WINAPI MimeInternat_Rfc1522Decode(IMimeInternational *iface, LPCSTR pszValue,
484                                                  LPSTR pszCharset,
485                                                  ULONG cchmax,
486                                                  LPSTR *ppszDecoded)
487 {
488     FIXME("stub\n");
489     return E_NOTIMPL;
490 }
491 
492 static HRESULT WINAPI MimeInternat_Rfc1522Encode(IMimeInternational *iface, LPCSTR pszValue,
493                                                  HCHARSET hCharset,
494                                                  LPSTR *ppszEncoded)
495 {
496     FIXME("stub\n");
497     return E_NOTIMPL;
498 }
499 
500 static IMimeInternationalVtbl mime_internat_vtbl =
501 {
502     MimeInternat_QueryInterface,
503     MimeInternat_AddRef,
504     MimeInternat_Release,
505     MimeInternat_SetDefaultCharset,
506     MimeInternat_GetDefaultCharset,
507     MimeInternat_GetCodePageCharset,
508     MimeInternat_FindCharset,
509     MimeInternat_GetCharsetInfo,
510     MimeInternat_GetCodePageInfo,
511     MimeInternat_CanConvertCodePages,
512     MimeInternat_DecodeHeader,
513     MimeInternat_EncodeHeader,
514     MimeInternat_ConvertBuffer,
515     MimeInternat_ConvertString,
516     MimeInternat_MLANG_ConvertInetReset,
517     MimeInternat_MLANG_ConvertInetString,
518     MimeInternat_Rfc1522Decode,
519     MimeInternat_Rfc1522Encode
520 };
521 
522 static internat *global_internat;
523 
524 HRESULT MimeInternational_Construct(IMimeInternational **internat)
525 {
526     global_internat = HeapAlloc(GetProcessHeap(), 0, sizeof(*global_internat));
527     global_internat->lpVtbl = &mime_internat_vtbl;
528     global_internat->refs = 0;
529     InitializeCriticalSection(&global_internat->cs);
530 
531     list_init(&global_internat->charsets);
532     global_internat->next_charset_handle = 0;
533     global_internat->default_charset = NULL;
534 
535     *internat = (IMimeInternational*)&global_internat->lpVtbl;
536 
537     IMimeInternational_AddRef(*internat);
538     return S_OK;
539 }
540 
541 HRESULT WINAPI MimeOleGetInternat(IMimeInternational **internat)
542 {
543     TRACE("(%p)\n", internat);
544 
545     *internat = (IMimeInternational *)&global_internat->lpVtbl;
546     IMimeInternational_AddRef(*internat);
547     return S_OK;
548 }
549 
550 HRESULT WINAPI MimeOleFindCharset(LPCSTR name, LPHCHARSET charset)
551 {
552     IMimeInternational *internat;
553     HRESULT hr;
554 
555     TRACE("(%s, %p)\n", debugstr_a(name), charset);
556 
557     hr = MimeOleGetInternat(&internat);
558     if(SUCCEEDED(hr))
559     {
560         hr = IMimeInternational_FindCharset(internat, name, charset);
561         IMimeInternational_Release(internat);
562     }
563     return hr;
564 }
565 
566 HRESULT WINAPI MimeOleGetCharsetInfo(HCHARSET hCharset, LPINETCSETINFO pCsetInfo)
567 {
568     IMimeInternational *internat;
569     HRESULT hr;
570 
571     TRACE("(%p, %p)\n", hCharset, pCsetInfo);
572 
573     hr = MimeOleGetInternat(&internat);
574     if(SUCCEEDED(hr))
575     {
576         hr = IMimeInternational_GetCharsetInfo(internat, hCharset, pCsetInfo);
577         IMimeInternational_Release(internat);
578     }
579     return hr;
580 }
581 
582 HRESULT WINAPI MimeOleGetDefaultCharset(LPHCHARSET charset)
583 {
584     IMimeInternational *internat;
585     HRESULT hr;
586 
587     TRACE("(%p)\n", charset);
588 
589     hr = MimeOleGetInternat(&internat);
590     if(SUCCEEDED(hr))
591     {
592         hr = IMimeInternational_GetDefaultCharset(internat, charset);
593         IMimeInternational_Release(internat);
594     }
595     return hr;
596 }
597 

~ [ source navigation ] ~ [ diff markup ] ~ [ identifier search ] ~ [ freetext search ] ~ [ file search ] ~

This page was automatically generated by the LXR engine.
Visit the LXR main site for more information.