diff options
Diffstat (limited to 'subversion/libsvn_subr/win32_xlate.c')
-rw-r--r-- | subversion/libsvn_subr/win32_xlate.c | 238 |
1 files changed, 238 insertions, 0 deletions
diff --git a/subversion/libsvn_subr/win32_xlate.c b/subversion/libsvn_subr/win32_xlate.c new file mode 100644 index 0000000..efe9c05 --- /dev/null +++ b/subversion/libsvn_subr/win32_xlate.c @@ -0,0 +1,238 @@ +/* + * win32_xlate.c : Windows xlate stuff. + * + * ==================================================================== + * Licensed to the Apache Software Foundation (ASF) under one + * or more contributor license agreements. See the NOTICE file + * distributed with this work for additional information + * regarding copyright ownership. The ASF licenses this file + * to you under the Apache License, Version 2.0 (the + * "License"); you may not use this file except in compliance + * with the License. You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, + * software distributed under the License is distributed on an + * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY + * KIND, either express or implied. See the License for the + * specific language governing permissions and limitations + * under the License. + * ==================================================================== + */ + +/* prevent "empty compilation unit" warning on e.g. UNIX */ +typedef int win32_xlate__dummy; + +#ifdef WIN32 + +/* Define _WIN32_DCOM for CoInitializeEx(). */ +#define _WIN32_DCOM + +/* We must include windows.h ourselves or apr.h includes it for us with + many ignore options set. Including Winsock is required to resolve IPv6 + compilation errors. APR_HAVE_IPV6 is only defined after including + apr.h, so we can't detect this case here. */ + +/* winsock2.h includes windows.h */ +#include <winsock2.h> +#include <Ws2tcpip.h> +#include <mlang.h> + +#include <apr.h> +#include <apr_errno.h> +#include <apr_portable.h> + +#include "svn_pools.h" +#include "svn_string.h" +#include "svn_utf.h" +#include "private/svn_atomic.h" + +#include "win32_xlate.h" + +static svn_atomic_t com_initialized = 0; + +/* Initializes COM and keeps COM available until process exit. + Implements svn_atomic__init_once init_func */ +static svn_error_t * +initialize_com(void *baton, apr_pool_t* pool) +{ + /* Try to initialize for apartment-threaded object concurrency. */ + HRESULT hr = CoInitializeEx(NULL, COINIT_APARTMENTTHREADED); + + if (hr == RPC_E_CHANGED_MODE) + { + /* COM already initalized for multi-threaded object concurrency. We are + neutral to object concurrency so try to initalize it in the same way + for us, to keep an handle open. */ + hr = CoInitializeEx(NULL, COINIT_MULTITHREADED); + } + + if (FAILED(hr)) + return svn_error_create(APR_EGENERAL, NULL, NULL); + + return SVN_NO_ERROR; +} + +typedef struct win32_xlate_t +{ + UINT from_page_id; + UINT to_page_id; +} win32_xlate_t; + +static apr_status_t +get_page_id_from_name(UINT *page_id_p, const char *page_name, apr_pool_t *pool) +{ + IMultiLanguage * mlang = NULL; + HRESULT hr; + MIMECSETINFO page_info; + WCHAR ucs2_page_name[128]; + svn_error_t *err; + + if (page_name == SVN_APR_DEFAULT_CHARSET) + { + *page_id_p = CP_ACP; + return APR_SUCCESS; + } + else if (page_name == SVN_APR_LOCALE_CHARSET) + { + *page_id_p = CP_THREAD_ACP; /* Valid on Windows 2000+ */ + return APR_SUCCESS; + } + else if (!strcmp(page_name, "UTF-8")) + { + *page_id_p = CP_UTF8; + return APR_SUCCESS; + } + + /* Use codepage identifier nnn if the codepage name is in the form + of "CPnnn". + We need this code since apr_os_locale_encoding() and svn_cmdline_init() + generates such codepage names even if they are not valid IANA charset + name. */ + if ((page_name[0] == 'c' || page_name[0] == 'C') + && (page_name[1] == 'p' || page_name[1] == 'P')) + { + *page_id_p = atoi(page_name + 2); + return APR_SUCCESS; + } + + err = svn_atomic__init_once(&com_initialized, initialize_com, NULL, pool); + + if (err) + { + svn_error_clear(err); + return APR_EGENERAL; + } + + hr = CoCreateInstance(&CLSID_CMultiLanguage, NULL, CLSCTX_INPROC_SERVER, + &IID_IMultiLanguage, (void **) &mlang); + + if (FAILED(hr)) + return APR_EGENERAL; + + /* Convert page name to wide string. */ + MultiByteToWideChar(CP_UTF8, 0, page_name, -1, ucs2_page_name, + sizeof(ucs2_page_name) / sizeof(ucs2_page_name[0])); + memset(&page_info, 0, sizeof(page_info)); + hr = mlang->lpVtbl->GetCharsetInfo(mlang, ucs2_page_name, &page_info); + if (FAILED(hr)) + { + mlang->lpVtbl->Release(mlang); + return APR_EINVAL; + } + + if (page_info.uiInternetEncoding) + *page_id_p = page_info.uiInternetEncoding; + else + *page_id_p = page_info.uiCodePage; + + mlang->lpVtbl->Release(mlang); + + return APR_SUCCESS; +} + +apr_status_t +svn_subr__win32_xlate_open(win32_xlate_t **xlate_p, const char *topage, + const char *frompage, apr_pool_t *pool) +{ + UINT from_page_id, to_page_id; + apr_status_t apr_err = APR_SUCCESS; + win32_xlate_t *xlate; + + apr_err = get_page_id_from_name(&to_page_id, topage, pool); + if (apr_err == APR_SUCCESS) + apr_err = get_page_id_from_name(&from_page_id, frompage, pool); + + if (apr_err == APR_SUCCESS) + { + xlate = apr_palloc(pool, sizeof(*xlate)); + xlate->from_page_id = from_page_id; + xlate->to_page_id = to_page_id; + + *xlate_p = xlate; + } + + return apr_err; +} + +apr_status_t +svn_subr__win32_xlate_to_stringbuf(win32_xlate_t *handle, + const char *src_data, + apr_size_t src_length, + svn_stringbuf_t **dest, + apr_pool_t *pool) +{ + WCHAR * wide_str; + int retval, wide_size; + + if (src_length == 0) + { + *dest = svn_stringbuf_create_empty(pool); + return APR_SUCCESS; + } + + retval = MultiByteToWideChar(handle->from_page_id, 0, src_data, src_length, + NULL, 0); + if (retval == 0) + return apr_get_os_error(); + + wide_size = retval; + + /* Allocate temporary buffer for small strings on stack instead of heap. */ + if (wide_size <= MAX_PATH) + { + wide_str = alloca(wide_size * sizeof(WCHAR)); + } + else + { + wide_str = apr_palloc(pool, wide_size * sizeof(WCHAR)); + } + + retval = MultiByteToWideChar(handle->from_page_id, 0, src_data, src_length, + wide_str, wide_size); + + if (retval == 0) + return apr_get_os_error(); + + retval = WideCharToMultiByte(handle->to_page_id, 0, wide_str, wide_size, + NULL, 0, NULL, NULL); + + if (retval == 0) + return apr_get_os_error(); + + /* Ensure that buffer is enough to hold result string and termination + character. */ + *dest = svn_stringbuf_create_ensure(retval + 1, pool); + (*dest)->len = retval; + + retval = WideCharToMultiByte(handle->to_page_id, 0, wide_str, wide_size, + (*dest)->data, (*dest)->len, NULL, NULL); + if (retval == 0) + return apr_get_os_error(); + + (*dest)->len = retval; + return APR_SUCCESS; +} + +#endif /* WIN32 */ |