diff --git a/CMakeLists.txt b/CMakeLists.txt index 4228d32ef..f0aa5aa5c 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -221,7 +221,10 @@ install(EXPORT celix NAMESPACE Celix:: DESTINATION share/celix/cmake FILE Target install_celix_targets(celix NAMESPACE Celix:: DESTINATION share/celix/cmake FILE CelixTargets COMPONENT cmake) #install celix cmake modules -install(DIRECTORY ${CMAKE_CURRENT_SOURCE_DIR}/cmake/Modules/ DESTINATION share/celix/cmake/Modules) +option(INSTALL_FIND_MODULES "Whether to install Find modules defined by Celix" ON) +if (INSTALL_FIND_MODULES) + install(DIRECTORY ${CMAKE_CURRENT_SOURCE_DIR}/cmake/Modules/ DESTINATION share/celix/cmake/Modules) +endif () install(DIRECTORY ${CMAKE_CURRENT_SOURCE_DIR}/cmake/cmake_celix/ DESTINATION share/celix/cmake/cmake_celix) file(GENERATE diff --git a/bundles/remote_services/remote_service_admin_dfi/src/export_registration_dfi.c b/bundles/remote_services/remote_service_admin_dfi/src/export_registration_dfi.c index ff45b0bea..54893b9d2 100644 --- a/bundles/remote_services/remote_service_admin_dfi/src/export_registration_dfi.c +++ b/bundles/remote_services/remote_service_admin_dfi/src/export_registration_dfi.c @@ -29,6 +29,8 @@ #include "dfi_utils.h" #include "remote_interceptors_handler.h" +#include + struct export_reference { endpoint_description_t *endpoint; //owner service_reference_pt reference; @@ -109,8 +111,7 @@ celix_status_t exportRegistration_create(celix_log_helper_t *helper, service_ref if (status == CELIX_SUCCESS) { /* Add the interface version as a property in the properties_map */ - char* intfVersion = NULL; - dynInterface_getVersionString(reg->intf, &intfVersion); + const char* intfVersion = dynInterface_getVersionString(reg->intf); const char *serviceVersion = celix_properties_get(endpoint->properties,(char*) CELIX_FRAMEWORK_SERVICE_VERSION, NULL); if (serviceVersion != NULL) { if(strcmp(serviceVersion,intfVersion)!=0){ @@ -190,8 +191,7 @@ celix_status_t exportRegistration_call(export_registration_t *export, char *data //printf("calling for '%s'\n"); if (export->logFile != NULL) { static int callCount = 0; - char *name = NULL; - dynInterface_getName(export->intf, &name); + const char *name = dynInterface_getName(export->intf); fprintf(export->logFile, "REMOTE CALL %i\n\tservice=%s\n\tservice_id=%s\n\trequest_payload=%s\n\trequest_response=%s\n\tstatus=%i\n", callCount, name, export->servId, data, response, status); fflush(export->logFile); callCount += 1; diff --git a/bundles/remote_services/remote_service_admin_dfi/src/import_registration_dfi.c b/bundles/remote_services/remote_service_admin_dfi/src/import_registration_dfi.c index 5cc0327e7..a0c55724a 100644 --- a/bundles/remote_services/remote_service_admin_dfi/src/import_registration_dfi.c +++ b/bundles/remote_services/remote_service_admin_dfi/src/import_registration_dfi.c @@ -20,7 +20,8 @@ #include #include #include -#include "version.h" +#include "celix_version.h" +#include "celix_stdlib_cleanup.h" #include "dyn_interface.h" #include "import_registration.h" #include "import_registration_dfi.h" @@ -32,7 +33,7 @@ struct import_registration { celix_bundle_context_t *context; endpoint_description_t * endpoint; //TODO owner? -> free when destroyed const char *classObject; //NOTE owned by endpoint - version_pt version; + celix_version_t* version; send_func_type send; void *sendHandle; @@ -87,8 +88,11 @@ celix_status_t importRegistration_create( remoteInterceptorsHandler_create(context, ®->interceptorsHandler); celixThreadMutex_create(®->proxiesMutex, NULL); - status = version_createVersionFromString((char*)serviceVersion,&(reg->version)); - + // serviceVersion == NULL is allowed, check TEST_F(RsaDfiTests, ImportService) + reg->version = celix_version_createVersionFromString(serviceVersion); + if (serviceVersion != NULL && reg->version == NULL) { + status = CELIX_ILLEGAL_ARGUMENT; + } reg->factorySvcId = -1; reg->factory.handle = reg; reg->factory.getService = importRegistration_getService; @@ -135,7 +139,7 @@ static void importRegistration_destroyCallback(void* data) { pthread_mutex_destroy(&import->proxiesMutex); if (import->version != NULL) { - version_destroy(import->version); + celix_version_destroy(import->version); } free(import); } @@ -230,21 +234,19 @@ static celix_status_t importRegistration_createProxy(import_registration_t *impo } /* Check if the imported service version is compatible with the one in the consumer descriptor */ - version_pt consumerVersion = NULL; + const celix_version_t* consumerVersion = dynInterface_getVersion(intf); bool isCompatible = false; - dynInterface_getVersion(intf,&consumerVersion); - version_isCompatible(consumerVersion,import->version,&isCompatible); + isCompatible = celix_version_isCompatible(consumerVersion,import->version); if(!isCompatible){ - char* cVerString = NULL; - char* pVerString = NULL; - version_toString(consumerVersion,&cVerString); - version_toString(import->version,&pVerString); - printf("Service version mismatch: consumer has %s, provider has %s. NOT creating proxy.\n",cVerString,pVerString); - dynInterface_destroy(intf); - free(cVerString); - free(pVerString); - status = CELIX_SERVICE_EXCEPTION; + celix_autofree char* cVerString = NULL; + celix_autofree char* pVerString = NULL; + cVerString = celix_version_toString(consumerVersion); + pVerString = import->version != NULL ? celix_version_toString(import->version) : NULL; + printf("Service version mismatch: consumer has %s, provider has %s. NOT creating proxy.\n", + cVerString,pVerString != NULL ? pVerString : "NA"); + dynInterface_destroy(intf); + status = CELIX_SERVICE_EXCEPTION; } struct service_proxy *proxy = NULL; @@ -268,8 +270,7 @@ static celix_status_t importRegistration_createProxy(import_registration_t *impo void **serv = proxy->service; serv[0] = import; - struct methods_head *list = NULL; - dynInterface_methods(proxy->intf, &list); + const struct methods_head* list = dynInterface_methods(proxy->intf); struct method_entry *entry = NULL; void (*fn)(void) = NULL; int index = 0; @@ -324,7 +325,8 @@ static void importRegistration_proxyFunc(void *userData, void *args[], void *ret int rc = 0; //printf("sending request\n"); celix_properties_t *metadata = NULL; - bool cont = remoteInterceptorHandler_invokePreProxyCall(import->interceptorsHandler, import->endpoint->properties, entry->name, &metadata); + bool cont = remoteInterceptorHandler_invokePreProxyCall(import->interceptorsHandler, import->endpoint->properties, + dynFunction_getName(entry->dynFunc), &metadata); if (cont) { status = import->send(import->sendHandle, import->endpoint, invokeRequest, metadata, &reply, &rc); //printf("request sended. got reply '%s' with status %i\n", reply, rc); @@ -343,7 +345,8 @@ static void importRegistration_proxyFunc(void *userData, void *args[], void *ret *(int *) returnVal = rc; } - remoteInterceptorHandler_invokePostProxyCall(import->interceptorsHandler, import->endpoint->properties, entry->name, metadata); + remoteInterceptorHandler_invokePostProxyCall(import->interceptorsHandler, import->endpoint->properties, + dynFunction_getName(entry->dynFunc), metadata); } else { *(int *) returnVal = CELIX_INTERCEPTOR_EXCEPTION; } diff --git a/bundles/remote_services/rsa_rpc_json/gtest/src/RsaJsonRpcUnitTestSuite.cc b/bundles/remote_services/rsa_rpc_json/gtest/src/RsaJsonRpcUnitTestSuite.cc index 47f906fb6..cdd3e5278 100644 --- a/bundles/remote_services/rsa_rpc_json/gtest/src/RsaJsonRpcUnitTestSuite.cc +++ b/bundles/remote_services/rsa_rpc_json/gtest/src/RsaJsonRpcUnitTestSuite.cc @@ -790,39 +790,6 @@ TEST_F(RsaJsonRpcEndPointUnitTestSuite, FailedToFindInterfaceDescriptor) { unsetenv("CELIX_FRAMEWORK_EXTENDER_PATH"); } -TEST_F(RsaJsonRpcEndPointUnitTestSuite, FailedToGetServiceVersionFromInterfaceDescriptor) { - celix_ei_expect_dynInterface_getVersionString(CELIX_EI_UNKNOWN_CALLER, 0, 1); - - auto endpoint = CreateEndpointDescription(rpcTestSvcId); - long svcId = -1L; - auto status = rsaJsonRpc_createEndpoint(jsonRpc.get(), endpoint, &svcId); - EXPECT_EQ(CELIX_SUCCESS, status); - - celix_bundleContext_waitForEvents(ctx.get());//wait for async endpoint creation - - unsigned int serialProtoId = GenerateSerialProtoId(); - celix_properties_t *metadata = celix_properties_create(); - celix_properties_setLong(metadata, "SerialProtocolId", serialProtoId); - - auto found = celix_bundleContext_useService(ctx.get(), RSA_REQUEST_HANDLER_SERVICE_NAME, metadata, [](void *handle, void *svc) { - celix_properties_t *metadata = static_cast< celix_properties_t *>(handle);//unused - auto reqHandler = static_cast(svc); - EXPECT_NE(nullptr, reqHandler); - struct iovec request{}; - request.iov_base = (char *)"{\n \"m\": \"test\",\n \"a\": []\n}"; - request.iov_len = strlen((char*)request.iov_base); - struct iovec reply{nullptr,0}; - EXPECT_EQ(CELIX_ILLEGAL_STATE, reqHandler->handleRequest(reqHandler->handle, metadata, &request, &reply)); - free(reply.iov_base); - }); - EXPECT_TRUE(found); - - celix_properties_destroy(metadata); - - rsaJsonRpc_destroyEndpoint(jsonRpc.get(), svcId); - endpointDescription_destroy(endpoint); -} - TEST_F(RsaJsonRpcEndPointUnitTestSuite, ServiceVersionMismatched) { auto endpoint = CreateEndpointDescription(rpcTestSvcId); celix_properties_set(endpoint->properties, CELIX_FRAMEWORK_SERVICE_VERSION, "2.0.0");//Its 1.0.0 in the interface descriptor diff --git a/bundles/remote_services/rsa_rpc_json/src/rsa_json_rpc_endpoint_impl.c b/bundles/remote_services/rsa_rpc_json/src/rsa_json_rpc_endpoint_impl.c index ee1dc5ef2..fd3e96951 100644 --- a/bundles/remote_services/rsa_rpc_json/src/rsa_json_rpc_endpoint_impl.c +++ b/bundles/remote_services/rsa_rpc_json/src/rsa_json_rpc_endpoint_impl.c @@ -29,6 +29,7 @@ #include #include #include +#include struct rsa_json_rpc_endpoint { celix_bundle_context_t* ctx; @@ -172,13 +173,7 @@ static void rsaJsonRpcEndpoint_addSvcWithOwner(void *handle, void *service, } // Check version - char *intfVersion = NULL; - int ret = dynInterface_getVersionString(intfType, &intfVersion); - if (ret != 0) { - celix_logHelper_logTssErrors(endpoint->logHelper, CELIX_LOG_LEVEL_ERROR); - celix_logHelper_error(endpoint->logHelper, "Endpoint: Error getting interface version from the descriptor for %s.", serviceName); - return; - } + const char* intfVersion = dynInterface_getVersionString(intfType); const char *serviceVersion = celix_properties_get(endpoint->endpointDesc->properties,CELIX_FRAMEWORK_SERVICE_VERSION, NULL); if (serviceVersion == NULL) { celix_logHelper_error(endpoint->logHelper, "Endpoint: Error getting service version for %s.", serviceName); diff --git a/bundles/remote_services/rsa_rpc_json/src/rsa_json_rpc_impl.c b/bundles/remote_services/rsa_rpc_json/src/rsa_json_rpc_impl.c index de86537f5..9bb19386c 100644 --- a/bundles/remote_services/rsa_rpc_json/src/rsa_json_rpc_impl.c +++ b/bundles/remote_services/rsa_rpc_json/src/rsa_json_rpc_impl.c @@ -36,6 +36,7 @@ #include #include #include +#include struct rsa_json_rpc { celix_bundle_context_t *ctx; diff --git a/bundles/remote_services/rsa_rpc_json/src/rsa_json_rpc_proxy_impl.c b/bundles/remote_services/rsa_rpc_json/src/rsa_json_rpc_proxy_impl.c index 83d505bcd..55914fc87 100644 --- a/bundles/remote_services/rsa_rpc_json/src/rsa_json_rpc_proxy_impl.c +++ b/bundles/remote_services/rsa_rpc_json/src/rsa_json_rpc_proxy_impl.c @@ -29,8 +29,9 @@ #include "celix_build_assert.h" #include "celix_long_hash_map.h" #include -#include #include +#include +#include struct rsa_json_rpc_proxy_factory { celix_bundle_context_t* ctx; @@ -212,7 +213,8 @@ static void rsaJsonRpcProxy_serviceFunc(void *userData, void *args[], void *retu int rc = jsonRpc_prepareInvokeRequest(entry->dynFunc, entry->id, args, &invokeRequest); if (rc != 0) { celix_logHelper_logTssErrors(proxyFactory->logHelper, CELIX_LOG_LEVEL_ERROR); - celix_logHelper_error(proxyFactory->logHelper, "Error preparing invoke request for %s", entry->name); + celix_logHelper_error(proxyFactory->logHelper, "Error preparing invoke request for %s", + dynFunction_getName(entry->dynFunc)); *(celix_status_t *)returnVal = CELIX_SERVICE_EXCEPTION; return; } @@ -220,14 +222,15 @@ static void rsaJsonRpcProxy_serviceFunc(void *userData, void *args[], void *retu struct iovec replyIovec = {NULL,0}; celix_properties_t *metadata = celix_properties_create(); if (metadata == NULL) { - celix_logHelper_error(proxyFactory->logHelper,"Error creating metadata for %s", entry->name); + celix_logHelper_error(proxyFactory->logHelper,"Error creating metadata for %s", + dynFunction_getName(entry->dynFunc)); free(invokeRequest); *(celix_status_t *)returnVal = CELIX_ENOMEM; return; } celix_properties_setLong(metadata, "SerialProtocolId", proxyFactory->serialProtoId); bool cont = remoteInterceptorHandler_invokePreProxyCall(proxyFactory->interceptorsHandler, - proxyFactory->endpointDesc->properties, entry->name, &metadata); + proxyFactory->endpointDesc->properties, dynFunction_getName(entry->dynFunc), &metadata); if (cont) { struct iovec requestIovec = {invokeRequest,strlen(invokeRequest) + 1}; struct rsa_request_sender_callback_data data= { @@ -246,7 +249,8 @@ static void rsaJsonRpcProxy_serviceFunc(void *userData, void *args[], void *retu if(retVal != 0) { status = CELIX_SERVICE_EXCEPTION; celix_logHelper_logTssErrors(proxyFactory->logHelper, CELIX_LOG_LEVEL_ERROR); - celix_logHelper_error(proxyFactory->logHelper, "Error handling reply for %s", entry->name); + celix_logHelper_error(proxyFactory->logHelper, "Error handling reply for %s", + dynFunction_getName(entry->dynFunc)); } else if (rsErrno != CELIX_SUCCESS) { //return the invocation error of remote service function status = rsErrno; @@ -259,7 +263,7 @@ static void rsaJsonRpcProxy_serviceFunc(void *userData, void *args[], void *retu celix_logHelper_error(proxyFactory->logHelper,"Service proxy send request failed. %d", status); } remoteInterceptorHandler_invokePostProxyCall(proxyFactory->interceptorsHandler, - proxyFactory->endpointDesc->properties, entry->name, metadata); + proxyFactory->endpointDesc->properties, dynFunction_getName(entry->dynFunc), metadata); } else { celix_logHelper_error(proxyFactory->logHelper, "%s has been intercepted.", proxyFactory->endpointDesc->serviceName); status = CELIX_INTERCEPTOR_EXCEPTION; @@ -317,9 +321,8 @@ static celix_status_t rsaJsonRpcProxy_create(rsa_json_rpc_proxy_factory_t *proxy celix_logHelper_error(proxyFactory->logHelper, "Proxy: Error converting service version type. %d.", status); return status; } - celix_version_t *consumerVersion = NULL; + const celix_version_t *consumerVersion = dynInterface_getVersion(intfType); bool isCompatible = false; - dynInterface_getVersion(intfType,&consumerVersion); isCompatible = celix_version_isCompatible(consumerVersion, providerVersion); if(!isCompatible){ celix_logHelper_error(proxyFactory->logHelper, "Proxy: Service version mismatch, consumer has %d.%d.%d, provider has %s.", @@ -336,15 +339,15 @@ static celix_status_t rsaJsonRpcProxy_create(rsa_json_rpc_proxy_factory_t *proxy } celix_autofree void **service = (void **)proxy->service; service[0] = proxy; - struct methods_head *list = NULL; - dynInterface_methods(intfType, &list); + const struct methods_head* list = dynInterface_methods(intfType); struct method_entry *entry = NULL; void (*fn)(void) = NULL; int index = 0; TAILQ_FOREACH(entry, list, entries) { int rc = dynFunction_createClosure(entry->dynFunc, rsaJsonRpcProxy_serviceFunc, entry, &fn); if (rc != 0) { - celix_logHelper_error(proxyFactory->logHelper, "Proxy: Failed to create closure for service function %s.", entry->name); + celix_logHelper_error(proxyFactory->logHelper, "Proxy: Failed to create closure for service function %s.", + dynFunction_getName(entry->dynFunc)); return CELIX_SERVICE_EXCEPTION; } service[++index] = fn; diff --git a/cmake/celix_project/CodeCoverage.cmake b/cmake/celix_project/CodeCoverage.cmake index aa33308e9..7615b38fc 100644 --- a/cmake/celix_project/CodeCoverage.cmake +++ b/cmake/celix_project/CodeCoverage.cmake @@ -70,8 +70,8 @@ IF(ENABLE_CODE_COVERAGE) # Setup compiler options ADD_DEFINITIONS(--coverage) - set(CMAKE_SHARED_LINKER_FLAGS "--coverage") - set(CMAKE_EXE_LINKER_FLAGS "--coverage") + set(CMAKE_SHARED_LINKER_FLAGS "${CMAKE_SHARED_LINKER_FLAGS} --coverage") + set(CMAKE_EXE_LINKER_FLAGS "${CMAKE_EXE_LINKER_FLAGS} --coverage") IF(NOT TARGET coverage) add_custom_target(coverage diff --git a/conanfile.py b/conanfile.py index b36bc30bb..60e05f719 100644 --- a/conanfile.py +++ b/conanfile.py @@ -47,6 +47,7 @@ class CelixConan(ConanFile): "enable_address_sanitizer": False, "enable_undefined_sanitizer": False, "enable_thread_sanitizer": False, + "install_find_modules": False, "build_all": False, "build_http_admin": False, "build_log_service": False, @@ -338,6 +339,8 @@ def generate(self): lst = [x.ref.name for x in self.requires.values()] if "mdnsresponder" in lst: tc.cache_variables["BUILD_ERROR_INJECTOR_MDNSRESPONDER"] = "ON" + if "jansson" in lst: + tc.cache_variables["BUILD_ERROR_INJECTOR_JANSSON"] = "ON" tc.cache_variables["CELIX_ERR_BUFFER_SIZE"] = str(self.options.celix_err_buffer_size) # tc.cache_variables["CMAKE_PROJECT_Celix_INCLUDE"] = os.path.join(self.build_folder, "conan_paths.cmake") # the following is workaround for https://github.com/conan-io/conan/issues/7192 diff --git a/libs/dfi/CMakeLists.txt b/libs/dfi/CMakeLists.txt index f82aca3a2..f6787d987 100644 --- a/libs/dfi/CMakeLists.txt +++ b/libs/dfi/CMakeLists.txt @@ -29,12 +29,14 @@ if (CELIX_DFI) src/dyn_message.c src/json_serializer.c src/json_rpc.c + src/dyn_descriptor.c ) add_library(dfi SHARED ${SOURCES}) target_include_directories(dfi PUBLIC $ - $) + $ + PRIVATE ${CMAKE_CURRENT_LIST_DIR}/src) target_link_libraries(dfi PRIVATE libffi::libffi) target_link_libraries(dfi PUBLIC jansson::jansson Celix::utils)##The public header file(dyn_interface.h) Celix::utils(celix_version.h) set_target_properties(dfi PROPERTIES @@ -60,6 +62,14 @@ if (CELIX_DFI) add_subdirectory(error_injector) endif () if (ENABLE_TESTING) + add_library(dfi_cut STATIC ${SOURCES}) + target_compile_definitions(dfi_cut PUBLIC CELIX_DFI_STATIC_DEFINE) + target_include_directories(dfi_cut PUBLIC + ${CMAKE_CURRENT_LIST_DIR}/src + ${CMAKE_CURRENT_LIST_DIR}/include + ${CMAKE_BINARY_DIR}/celix/gen/includes/dfi + ) + target_link_libraries(dfi_cut PUBLIC libffi::libffi jansson::jansson Celix::utils) add_subdirectory(gtest) endif(ENABLE_TESTING) endif (CELIX_DFI) diff --git a/libs/dfi/error_injector/CMakeLists.txt b/libs/dfi/error_injector/CMakeLists.txt index f6d2a84c9..bc63f6796 100644 --- a/libs/dfi/error_injector/CMakeLists.txt +++ b/libs/dfi/error_injector/CMakeLists.txt @@ -15,4 +15,5 @@ # specific language governing permissions and limitations # under the License. -add_subdirectory(dfi) \ No newline at end of file +add_subdirectory(dfi) +add_subdirectory(ffi) diff --git a/libs/dfi/error_injector/dfi/CMakeLists.txt b/libs/dfi/error_injector/dfi/CMakeLists.txt index 23608c884..34fd1713a 100644 --- a/libs/dfi/error_injector/dfi/CMakeLists.txt +++ b/libs/dfi/error_injector/dfi/CMakeLists.txt @@ -23,6 +23,5 @@ target_link_libraries(dfi_ei PUBLIC Celix::error_injector Celix::dfi) target_link_options(dfi_ei INTERFACE LINKER:--wrap,dynFunction_createClosure LINKER:--wrap,jsonRpc_prepareInvokeRequest - LINKER:--wrap,dynInterface_getVersionString ) add_library(Celix::dfi_ei ALIAS dfi_ei) diff --git a/libs/dfi/error_injector/dfi/include/dfi_ei.h b/libs/dfi/error_injector/dfi/include/dfi_ei.h index 140d922d1..cdab04f90 100644 --- a/libs/dfi/error_injector/dfi/include/dfi_ei.h +++ b/libs/dfi/error_injector/dfi/include/dfi_ei.h @@ -28,8 +28,6 @@ CELIX_EI_DECLARE(dynFunction_createClosure, int); CELIX_EI_DECLARE(jsonRpc_prepareInvokeRequest, int); -CELIX_EI_DECLARE(dynInterface_getVersionString, int); - #ifdef __cplusplus } #endif diff --git a/libs/dfi/error_injector/dfi/src/dfi_ei.cc b/libs/dfi/error_injector/dfi/src/dfi_ei.cc index b37f6060b..4950d7f60 100644 --- a/libs/dfi/error_injector/dfi/src/dfi_ei.cc +++ b/libs/dfi/error_injector/dfi/src/dfi_ei.cc @@ -35,11 +35,4 @@ int __wrap_jsonRpc_prepareInvokeRequest(dyn_function_type *func, const char *id, return __real_jsonRpc_prepareInvokeRequest(func, id, args, out); } -int __real_dynInterface_getVersionString(dyn_interface_type *dynInterface, char **out); -CELIX_EI_DEFINE(dynInterface_getVersionString, int) -int __wrap_dynInterface_getVersionString(dyn_interface_type *dynInterface, char **out) { - CELIX_EI_IMPL(dynInterface_getVersionString); - return __real_dynInterface_getVersionString(dynInterface, out); -} - } \ No newline at end of file diff --git a/libs/dfi/error_injector/ffi/CMakeLists.txt b/libs/dfi/error_injector/ffi/CMakeLists.txt new file mode 100644 index 000000000..c79a429a1 --- /dev/null +++ b/libs/dfi/error_injector/ffi/CMakeLists.txt @@ -0,0 +1,28 @@ +# 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. + +add_library(ffi_ei STATIC src/ffi_ei.cc) + +target_include_directories(ffi_ei PUBLIC include) +target_link_libraries(ffi_ei PUBLIC Celix::error_injector libffi::libffi) + +target_link_options(ffi_ei INTERFACE + LINKER:--wrap,ffi_prep_cif + LINKER:--wrap,ffi_closure_alloc + LINKER:--wrap,ffi_prep_closure_loc +) +add_library(Celix::ffi_ei ALIAS ffi_ei) diff --git a/libs/dfi/error_injector/ffi/include/ffi_ei.h b/libs/dfi/error_injector/ffi/include/ffi_ei.h new file mode 100644 index 000000000..f1dd1fb0b --- /dev/null +++ b/libs/dfi/error_injector/ffi/include/ffi_ei.h @@ -0,0 +1,37 @@ +/* + * 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. + */ +#ifndef CELIX_FFI_EI_H +#define CELIX_FFI_EI_H +#ifdef __cplusplus +extern "C" { +#endif + +#include "celix_error_injector.h" +#include + +CELIX_EI_DECLARE(ffi_prep_cif, ffi_status); + +CELIX_EI_DECLARE(ffi_closure_alloc, void*); + +CELIX_EI_DECLARE(ffi_prep_closure_loc, ffi_status); + +#ifdef __cplusplus +} +#endif +#endif //CELIX_FFI_EI_H diff --git a/libs/dfi/error_injector/ffi/src/ffi_ei.cc b/libs/dfi/error_injector/ffi/src/ffi_ei.cc new file mode 100644 index 000000000..f34dc5b69 --- /dev/null +++ b/libs/dfi/error_injector/ffi/src/ffi_ei.cc @@ -0,0 +1,44 @@ +/* + * 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. + */ + +#include "ffi_ei.h" + +extern "C" { +ffi_status __real_ffi_prep_cif(ffi_cif *cif, ffi_abi abi, unsigned int nargs, ffi_type *rtype, ffi_type **atypes); +CELIX_EI_DEFINE(ffi_prep_cif, ffi_status) +ffi_status __wrap_ffi_prep_cif(ffi_cif *cif, ffi_abi abi, unsigned int nargs, ffi_type *rtype, ffi_type **atypes) { + CELIX_EI_IMPL(ffi_prep_cif); + return __real_ffi_prep_cif(cif, abi, nargs, rtype, atypes); +} + +void *__real_ffi_closure_alloc(size_t size, void **code); +CELIX_EI_DEFINE(ffi_closure_alloc, void*) +void *__wrap_ffi_closure_alloc(size_t size, void **code) { + CELIX_EI_IMPL(ffi_closure_alloc); + return __real_ffi_closure_alloc(size, code); +} + +ffi_status __real_ffi_prep_closure_loc(ffi_closure *closure, ffi_cif *cif, void (*fun)(ffi_cif *, void *, void **, void *), void *user_data, void *codeloc); +CELIX_EI_DEFINE(ffi_prep_closure_loc, ffi_status) +ffi_status __wrap_ffi_prep_closure_loc(ffi_closure *closure, ffi_cif *cif, void (*fun)(ffi_cif *, void *, void **, void *), void *user_data, void *codeloc) { + CELIX_EI_IMPL(ffi_prep_closure_loc); + return __real_ffi_prep_closure_loc(closure, cif, fun, user_data, codeloc); +} + +} \ No newline at end of file diff --git a/libs/dfi/gtest/CMakeLists.txt b/libs/dfi/gtest/CMakeLists.txt index e0206df13..7e23890f9 100644 --- a/libs/dfi/gtest/CMakeLists.txt +++ b/libs/dfi/gtest/CMakeLists.txt @@ -18,18 +18,20 @@ add_executable(test_dfi src/dyn_example_functions.c - src/dyn_type_tests.cpp + src/dyn_type_tests.cpp src/dyn_function_tests.cpp src/dyn_closure_tests.cpp src/dyn_interface_tests.cpp src/dyn_message_tests.cpp src/json_serializer_tests.cpp src/json_rpc_tests.cpp + src/dyn_common_tests.cc + src/json_rpc_test.c ) target_include_directories(test_dfi PRIVATE ${CMAKE_CURRENT_LIST_DIR}/../src) -target_link_libraries(test_dfi PRIVATE Celix::dfi Celix::utils libffi::libffi jansson::jansson GTest::gtest GTest::gtest_main) +target_link_libraries(test_dfi PRIVATE dfi_cut Celix::utils libffi::libffi jansson::jansson GTest::gtest GTest::gtest_main) celix_deprecated_utils_headers(test_dfi) file(COPY ${CMAKE_CURRENT_LIST_DIR}/descriptors DESTINATION ${CMAKE_CURRENT_BINARY_DIR}) @@ -37,3 +39,27 @@ file(COPY ${CMAKE_CURRENT_LIST_DIR}/descriptors DESTINATION ${CMAKE_CURRENT_BINA add_test(NAME run_test_dfi COMMAND test_dfi) setup_target_for_coverage(test_dfi SCAN_DIR ..) +if (EI_TESTS) + add_executable(test_dfi_with_ei + src/dyn_interface_ei_tests.cc + src/dyn_common_ei_tests.cc + src/dyn_type_ei_tests.cc + src/dyn_message_ei_tests.cc + src/dyn_function_ei_tests.cc + src/json_serializer_ei_tests.cc + src/json_rpc_ei_tests.cc + src/json_rpc_test.c + ) + target_link_libraries(test_dfi_with_ei PRIVATE + dfi_cut + Celix::malloc_ei + Celix::stdio_ei + Celix::string_ei + Celix::ffi_ei + Celix::asprintf_ei + Celix::jansson_ei + GTest::gtest GTest::gtest_main + ) + add_test(NAME run_test_dfi_with_ei COMMAND test_dfi_with_ei) + setup_target_for_coverage(test_dfi_with_ei SCAN_DIR ..) +endif () diff --git a/libs/dfi/gtest/descriptors/example6.descriptor b/libs/dfi/gtest/descriptors/example6.descriptor index 4ce4e9357..5f9a06b63 100644 --- a/libs/dfi/gtest/descriptors/example6.descriptor +++ b/libs/dfi/gtest/descriptors/example6.descriptor @@ -1,9 +1,9 @@ :header type=interface -name=example6 +name=calculator version=1.0.0 :annotations +classname=org.example.Calculator :types -CptData={Dt[D#v1=1;#v2=2;E d t s e} +StatsResult={DDD[D average min max input} :methods -compatibility=cpt(#am=handle;PLCptData;#am=out;*LCptData;)N diff --git a/libs/dfi/gtest/descriptors/example7.descriptor b/libs/dfi/gtest/descriptors/example7.descriptor new file mode 100644 index 000000000..6729be3af --- /dev/null +++ b/libs/dfi/gtest/descriptors/example7.descriptor @@ -0,0 +1,13 @@ +:header +type=interface +name=calculator +version=1.0.0 +:annotations +classname=org.example.Calculator +:types +StatsResult={DDD[D average min max input} +:methods +add(DD)D=add(#am=handle;PDD#am=pre;TOut=*D;lOut;)N +sub(DD)D=sub(#am=handle;PDD*#am=pre;D)N +sqrt(D)D=sqrt(#am=handle;PD*#am=pre;D)N +stats([D)LStatsResult;=stats(#am=handle;P[D#am=out;TOut=*LStatsResult;;lOut;)N diff --git a/libs/dfi/gtest/descriptors/invalids/garbage.descriptor b/libs/dfi/gtest/descriptors/invalids/garbage.descriptor new file mode 100644 index 000000000..88c65efb1 --- /dev/null +++ b/libs/dfi/gtest/descriptors/invalids/garbage.descriptor @@ -0,0 +1 @@ +this is garbage diff --git a/libs/dfi/gtest/descriptors/invalids/invalidExtraSection.descriptor b/libs/dfi/gtest/descriptors/invalids/invalidExtraSection.descriptor new file mode 100644 index 000000000..bfd48d66e --- /dev/null +++ b/libs/dfi/gtest/descriptors/invalids/invalidExtraSection.descriptor @@ -0,0 +1,15 @@ +:header +type=interface +name=calculator +version=1.0.0 +:annotations +classname=org.example.Calculator +:types +StatsResult={DDD[D average min max input} +:methods +add(DD)D=add(#am=handle;PDD#am=pre;*D)N +sub(DD)D=sub(#am=handle;PDD*#am=pre;D)N +sqrt(D)D=sqrt(#am=handle;PD*#am=pre;D)N +stats([D)LStatsResult;=stats(#am=handle;P[D#am=out;*LStatsResult;)N +:invalidSection +invalidKey=invalidValue diff --git a/libs/dfi/gtest/descriptors/invalids/invalidInterfaceAnnotations.descriptor b/libs/dfi/gtest/descriptors/invalids/invalidInterfaceAnnotations.descriptor new file mode 100644 index 000000000..a5ecd55e4 --- /dev/null +++ b/libs/dfi/gtest/descriptors/invalids/invalidInterfaceAnnotations.descriptor @@ -0,0 +1,13 @@ +:header +type=interface +name=calculator +version=1.0.0 +:annotations +=org.example.Calculator +:types +StatsResult={DDD[D average min max input} +:methods +add(DD)D=add(#am=handle;PDD#am=pre;*D)N +sub(DD)D=sub(#am=handle;PDD*#am=pre;D)N +sqrt(D)D=sqrt(#am=handle;PD*#am=pre;D)N +stats([D)LStatsResult;=stats(#am=handle;P[D#am=out;*LStatsResult;)N diff --git a/libs/dfi/gtest/descriptors/invalids/invalidMetaType.descriptor b/libs/dfi/gtest/descriptors/invalids/invalidMetaType.descriptor index 2d17c4799..365c6dded 100644 --- a/libs/dfi/gtest/descriptors/invalids/invalidMetaType.descriptor +++ b/libs/dfi/gtest/descriptors/invalids/invalidMetaType.descriptor @@ -5,4 +5,4 @@ version=1.0.0 :annotations classname=org.example.Calculator :methods -add(DD)D=add(#am=invalid;PDD#am=pre;*D)N +add(DD)D=add(#am=handle;P#am=invalid;PDD#am=pre;*D)N diff --git a/libs/dfi/gtest/descriptors/invalids/invalidMethodMissingEquality.descriptor b/libs/dfi/gtest/descriptors/invalids/invalidMethodMissingEquality.descriptor new file mode 100644 index 000000000..c7b369e2f --- /dev/null +++ b/libs/dfi/gtest/descriptors/invalids/invalidMethodMissingEquality.descriptor @@ -0,0 +1,8 @@ +:header +type=interface +name=calculator +version=1.0.0 +:annotations +classname=org.example.Calculator +:methods +add(DD)D add(#am=handle;PDD#am=pre;*D)N diff --git a/libs/dfi/gtest/descriptors/invalids/invalidMethodMissingFunctionName.descriptor b/libs/dfi/gtest/descriptors/invalids/invalidMethodMissingFunctionName.descriptor new file mode 100644 index 000000000..092e5781f --- /dev/null +++ b/libs/dfi/gtest/descriptors/invalids/invalidMethodMissingFunctionName.descriptor @@ -0,0 +1,8 @@ +:header +type=interface +name=calculator +version=1.0.0 +:annotations +classname=org.example.Calculator +:methods +add(DD)D=(#am=handle;PDD#am=pre;*D)N diff --git a/libs/dfi/gtest/descriptors/invalids/invalidMethodMissingId.descriptor b/libs/dfi/gtest/descriptors/invalids/invalidMethodMissingId.descriptor new file mode 100644 index 000000000..62bb20b26 --- /dev/null +++ b/libs/dfi/gtest/descriptors/invalids/invalidMethodMissingId.descriptor @@ -0,0 +1,8 @@ +:header +type=interface +name=calculator +version=1.0.0 +:annotations +classname=org.example.Calculator +:methods +=add(#am=handle;PDD#am=pre;*D)N diff --git a/libs/dfi/gtest/descriptors/invalids/invalidMsgMissingName.descriptor b/libs/dfi/gtest/descriptors/invalids/invalidMsgMissingName.descriptor new file mode 100644 index 000000000..9010cb841 --- /dev/null +++ b/libs/dfi/gtest/descriptors/invalids/invalidMsgMissingName.descriptor @@ -0,0 +1,9 @@ +:header +type=message +version=1.0.0 +:annotations +classname=org.example.PointOfInterest +:types +location={DD lat long} +:message +{llocation;tt location name description} diff --git a/libs/dfi/gtest/descriptors/invalids/invalidMsgMissingNewline.descriptor b/libs/dfi/gtest/descriptors/invalids/invalidMsgMissingNewline.descriptor new file mode 100644 index 000000000..4f2cf8788 --- /dev/null +++ b/libs/dfi/gtest/descriptors/invalids/invalidMsgMissingNewline.descriptor @@ -0,0 +1,10 @@ +:header +type=message +name=poi +version=1.0.0 +:annotations +classname=org.example.PointOfInterest +:types +location={DD lat long} +:message +{llocation;tt location name description} \ No newline at end of file diff --git a/libs/dfi/gtest/descriptors/invalids/invalidMsgMissingVersion.descriptor b/libs/dfi/gtest/descriptors/invalids/invalidMsgMissingVersion.descriptor index 1fd3595ac..ae9b13158 100644 --- a/libs/dfi/gtest/descriptors/invalids/invalidMsgMissingVersion.descriptor +++ b/libs/dfi/gtest/descriptors/invalids/invalidMsgMissingVersion.descriptor @@ -1,7 +1,6 @@ -:head +:header type=message name=poi -version=1.0.0 :annotations classname=org.example.PointOfInterest :types diff --git a/libs/dfi/gtest/descriptors/invalids/invalidMsgType.descriptor b/libs/dfi/gtest/descriptors/invalids/invalidMsgType.descriptor new file mode 100644 index 000000000..cf31bf579 --- /dev/null +++ b/libs/dfi/gtest/descriptors/invalids/invalidMsgType.descriptor @@ -0,0 +1,10 @@ +:header +type=message +name=poi +version=1.0.0 +:annotations +classname=org.example.PointOfInterest +:types +location={DD lat long} +:message +{llocation;ttX location name description} diff --git a/libs/dfi/gtest/descriptors/invalids/invalidTypeMissingEquality.descriptor b/libs/dfi/gtest/descriptors/invalids/invalidTypeMissingEquality.descriptor new file mode 100644 index 000000000..ac567a8d9 --- /dev/null +++ b/libs/dfi/gtest/descriptors/invalids/invalidTypeMissingEquality.descriptor @@ -0,0 +1,10 @@ +:header +type=interface +name=calculator +version=1.0.0 +:annotations +classname=org.example.Calculator +:types +StatsResult {DDD[D average min max input} +:methods +stats([D)LStatsResult;=stats(#am=handle;P[D#am=out;*LStatsResult;)N diff --git a/libs/dfi/gtest/descriptors/invalids/invalidTypeUnrecognizedSimpleType.descriptor b/libs/dfi/gtest/descriptors/invalids/invalidTypeUnrecognizedSimpleType.descriptor new file mode 100644 index 000000000..4bdce8011 --- /dev/null +++ b/libs/dfi/gtest/descriptors/invalids/invalidTypeUnrecognizedSimpleType.descriptor @@ -0,0 +1,10 @@ +:header +type=interface +name=calculator +version=1.0.0 +:annotations +classname=org.example.Calculator +:types +StatsResult={X x} +:methods +stats([D)LStatsResult;=stats(#am=handle;P[D#am=out;*LStatsResult;)N diff --git a/libs/dfi/gtest/descriptors/invalids/methodMissingHandle.descriptor b/libs/dfi/gtest/descriptors/invalids/methodMissingHandle.descriptor new file mode 100644 index 000000000..50ac049d7 --- /dev/null +++ b/libs/dfi/gtest/descriptors/invalids/methodMissingHandle.descriptor @@ -0,0 +1,13 @@ +:header +type=interface +name=calculator +version=1.0.0 +:annotations +classname=org.example.Calculator +:types +StatsResult={DDD[D average min max input} +:methods +add(DD)D=add(DD#am=pre;*D)N +sub(DD)D=sub(#am=handle;PDD*#am=pre;D)N +sqrt(D)D=sqrt(#am=handle;PD*#am=pre;D)N +stats([D)LStatsResult;=stats(#am=handle;P[D#am=out;*LStatsResult;)N diff --git a/libs/dfi/gtest/descriptors/invalids/methodWithMultipleHandles.descriptor b/libs/dfi/gtest/descriptors/invalids/methodWithMultipleHandles.descriptor new file mode 100644 index 000000000..1bbf1abfb --- /dev/null +++ b/libs/dfi/gtest/descriptors/invalids/methodWithMultipleHandles.descriptor @@ -0,0 +1,10 @@ +:header +type=interface +name=calculator +version=1.0.0 +:annotations +classname=org.example.Calculator +:types +StatsResult={DDD[D average min max input} +:methods +add(DD)D=add(#am=handle;P#am=handle;PDD#am=pre;*D)N diff --git a/libs/dfi/gtest/descriptors/invalids/methodWithTooManyArgs.descriptor b/libs/dfi/gtest/descriptors/invalids/methodWithTooManyArgs.descriptor new file mode 100644 index 000000000..a89a5b89e --- /dev/null +++ b/libs/dfi/gtest/descriptors/invalids/methodWithTooManyArgs.descriptor @@ -0,0 +1,10 @@ +:header +type=interface +name=calculator +version=1.0.0 +:annotations +classname=org.example.Calculator +:types +StatsResult={DDD[D average min max input} +:methods +add(DDDDDDDDDDDDDDD)D=add(#am=handle;PDDDDDDDDDDDDDDD#am=pre;*D)N diff --git a/libs/dfi/gtest/descriptors/invalids/methodWithoutArguments.descriptor b/libs/dfi/gtest/descriptors/invalids/methodWithoutArguments.descriptor new file mode 100644 index 000000000..a83b54b51 --- /dev/null +++ b/libs/dfi/gtest/descriptors/invalids/methodWithoutArguments.descriptor @@ -0,0 +1,13 @@ +:header +type=interface +name=calculator +version=1.0.0 +:annotations +classname=org.example.Calculator +:types +StatsResult={DDD[D average min max input} +:methods +add(DD)D=add()N +sub(DD)D=sub(#am=handle;PDD*#am=pre;D)N +sqrt(D)D=sqrt(#am=handle;PD*#am=pre;D)N +stats([D)LStatsResult;=stats(#am=handle;P[D#am=out;*LStatsResult;)N diff --git a/libs/dfi/gtest/descriptors/invalids/multiOutArgs.descriptor b/libs/dfi/gtest/descriptors/invalids/multiOutArgs.descriptor index 4a7831ac3..be5d618ff 100644 --- a/libs/dfi/gtest/descriptors/invalids/multiOutArgs.descriptor +++ b/libs/dfi/gtest/descriptors/invalids/multiOutArgs.descriptor @@ -5,5 +5,4 @@ version=1.0.0 :annotations :types :methods -multiPreOut(V)Di=multiPreOut(#am=handle;P#am=pre;*D#am=pre;*i)N multiOut(V)Di=multiOut(#am=handle;P#am=out;**D#am=out;**i)N diff --git a/libs/dfi/gtest/descriptors/invalids/multiPreOutArgs.descriptor b/libs/dfi/gtest/descriptors/invalids/multiPreOutArgs.descriptor new file mode 100644 index 000000000..a34940988 --- /dev/null +++ b/libs/dfi/gtest/descriptors/invalids/multiPreOutArgs.descriptor @@ -0,0 +1,8 @@ +:header +type=interface +name=example5 +version=1.0.0 +:annotations +:types +:methods +multiPreOut(V)Di=multiPreOut(#am=handle;P#am=pre;*D#am=pre;*i)N diff --git a/libs/dfi/gtest/descriptors/invalids/noName.descriptor b/libs/dfi/gtest/descriptors/invalids/noName.descriptor new file mode 100644 index 000000000..819e0ede3 --- /dev/null +++ b/libs/dfi/gtest/descriptors/invalids/noName.descriptor @@ -0,0 +1,12 @@ +:header +type=interface +version=1.0.0 +:annotations +classname=org.example.Calculator +:types +StatsResult={DDD[D average min max input} +:methods +add(DD)D=add(#am=handle;PDD#am=pre;*D)N +sub(DD)D=sub(#am=handle;PDD*#am=pre;D)N +sqrt(D)D=sqrt(#am=handle;PD*#am=pre;D)N +stats([D)LStatsResult;=stats(#am=handle;P[D#am=out;*LStatsResult;)N diff --git a/libs/dfi/gtest/descriptors/invalids/noType.descriptor b/libs/dfi/gtest/descriptors/invalids/noType.descriptor new file mode 100644 index 000000000..60933df40 --- /dev/null +++ b/libs/dfi/gtest/descriptors/invalids/noType.descriptor @@ -0,0 +1,12 @@ +:header +name=calculator +version=1.0.0 +:annotations +classname=org.example.Calculator +:types +StatsResult={DDD[D average min max input} +:methods +add(DD)D=add(#am=handle;PDD#am=pre;*D)N +sub(DD)D=sub(#am=handle;PDD*#am=pre;D)N +sqrt(D)D=sqrt(#am=handle;PD*#am=pre;D)N +stats([D)LStatsResult;=stats(#am=handle;P[D#am=out;*LStatsResult;)N diff --git a/libs/dfi/gtest/descriptors/invalids/noTypeName.descriptor b/libs/dfi/gtest/descriptors/invalids/noTypeName.descriptor new file mode 100644 index 000000000..9d6825d59 --- /dev/null +++ b/libs/dfi/gtest/descriptors/invalids/noTypeName.descriptor @@ -0,0 +1,10 @@ +:header +type=interface +name=calculator +version=1.0.0 +:annotations +classname=org.example.Calculator +:types +={DDD[D average min max input} +:methods +stats([D)LStatsResult;=stats(#am=handle;P[D#am=out;*LStatsResult;)N diff --git a/libs/dfi/gtest/descriptors/invalids/outArgAtWrongPosition.descriptor b/libs/dfi/gtest/descriptors/invalids/outArgAtWrongPosition.descriptor new file mode 100644 index 000000000..19205067c --- /dev/null +++ b/libs/dfi/gtest/descriptors/invalids/outArgAtWrongPosition.descriptor @@ -0,0 +1,8 @@ +:header +type=interface +name=example5 +version=1.0.0 +:annotations +:types +:methods +outWrongPos(V)Di=outWrongPos(#am=handle;P#am=out;**i**D)N diff --git a/libs/dfi/gtest/src/dyn_closure_tests.cpp b/libs/dfi/gtest/src/dyn_closure_tests.cpp index 819a4aa0f..54bea43df 100644 --- a/libs/dfi/gtest/src/dyn_closure_tests.cpp +++ b/libs/dfi/gtest/src/dyn_closure_tests.cpp @@ -35,16 +35,6 @@ extern "C" { static int g_count; -static void stdLog(void*, int level, const char *file, int line, const char *msg, ...) { - va_list ap; - const char *levels[5] = {"NIL", "ERROR", "WARNING", "INFO", "DEBUG"}; - fprintf(stderr, "%s: FILE:%s, LINE:%i, MSG:",levels[level], file, line); - va_start(ap, msg); - vfprintf(stderr, msg, ap); - fprintf(stderr, "\n"); - va_end(ap); -} - #define EXAMPLE1_DESCRIPTOR "example(III)I" static void example1_binding(void*, void* args[], void *out) { int32_t a = *((int32_t *)args[0]); @@ -151,10 +141,6 @@ static void tests() { class DynClosureTests : public ::testing::Test { public: DynClosureTests() { - int lvl = 1; - dynFunction_logSetup(stdLog, NULL, lvl); - dynType_logSetup(stdLog, NULL, lvl); - dynCommon_logSetup(stdLog, NULL, lvl); g_count = 0; } ~DynClosureTests() override { diff --git a/libs/dfi/gtest/src/dyn_common_ei_tests.cc b/libs/dfi/gtest/src/dyn_common_ei_tests.cc new file mode 100644 index 000000000..8e92e631c --- /dev/null +++ b/libs/dfi/gtest/src/dyn_common_ei_tests.cc @@ -0,0 +1,84 @@ +/* + * 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. + */ + +#include "dyn_common.h" +#include "celix_err.h" +#include "malloc_ei.h" +#include "stdio_ei.h" + +#include + +class DynCommonErrorInjectionTestSuite : public ::testing::Test { +protected: + char* result{nullptr}; + FILE* stream{nullptr}; + + DynCommonErrorInjectionTestSuite() = default; + + ~DynCommonErrorInjectionTestSuite() override { + celix_ei_expect_calloc(nullptr, 0, nullptr); + celix_ei_expect_fclose(nullptr, 0, 0); + celix_ei_expect_fputc(nullptr, 0, 0); + celix_ei_expect_open_memstream(nullptr, 0, nullptr); + if (stream != nullptr) { + fclose(stream); + } + if (result != nullptr) { + free(result); + } + } + // delete other constructors and assign operators + DynCommonErrorInjectionTestSuite(DynCommonErrorInjectionTestSuite const&) = delete; + DynCommonErrorInjectionTestSuite(DynCommonErrorInjectionTestSuite&&) = delete; + DynCommonErrorInjectionTestSuite& operator=(DynCommonErrorInjectionTestSuite const&) = delete; + DynCommonErrorInjectionTestSuite& operator=(DynCommonErrorInjectionTestSuite&&) = delete; +}; + +TEST_F(DynCommonErrorInjectionTestSuite, ParseNameErrors) { + stream = fmemopen((void *) "valid_name", 10, "r"); + + // not enough memory for name + celix_ei_expect_open_memstream((void *) dynCommon_parseName, 1, nullptr); + ASSERT_EQ(dynCommon_parseName(stream, &result), 1); + std::string msg = "Error creating mem stream for name. "; + msg += strerror(ENOMEM); + ASSERT_STREQ(msg.c_str(), celix_err_popLastError()); + + // fail to put character into name + celix_ei_expect_fputc((void *) dynCommon_parseName, 1, EOF); + ASSERT_EQ(dynCommon_parseName(stream, &result), 1); + ASSERT_STREQ("Error writing to mem stream for name.", celix_err_popLastError()); + + // fail to close name stream + celix_ei_expect_fclose((void *) dynCommon_parseName, 1, EOF); + ASSERT_EQ(dynCommon_parseName(stream, &result), 1); + msg = "Error closing mem stream for name. "; + msg += strerror(ENOSPC); + ASSERT_STREQ(msg.c_str(), celix_err_popLastError()); +} + +TEST_F(DynCommonErrorInjectionTestSuite, ParseNameValueSectionErrors) { + stream = fmemopen((void*)"name1=value1\nname2=value2\n", 26, "r"); + struct namvals_head head; + TAILQ_INIT(&head); + // not enough memory for namval_entry when parsing name value section + celix_ei_expect_calloc((void *) dynCommon_parseNameValueSection, 0, nullptr, 1); + ASSERT_EQ(dynCommon_parseNameValueSection(stream, &head), 1); + ASSERT_STREQ("Error allocating memory for namval entry", celix_err_popLastError()); +} diff --git a/libs/dfi/gtest/src/dyn_common_tests.cc b/libs/dfi/gtest/src/dyn_common_tests.cc new file mode 100644 index 000000000..2848ab19a --- /dev/null +++ b/libs/dfi/gtest/src/dyn_common_tests.cc @@ -0,0 +1,149 @@ +/* + * 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. + */ + +#include "dyn_common.h" +#include "celix_err.h" +#include +#include +#include + +class DynCommonTests : public ::testing::Test { +protected: + char* result{nullptr}; + FILE* stream{nullptr}; + DynCommonTests() { + } + ~DynCommonTests() override { + if (stream != nullptr) { + fclose(stream); + } + if (result != nullptr) { + free(result); + } + } + // delete other constructors and assign operators + DynCommonTests(DynCommonTests const&) = delete; + DynCommonTests(DynCommonTests&&) = delete; + DynCommonTests& operator=(DynCommonTests const&) = delete; + DynCommonTests& operator=(DynCommonTests&&) = delete; +}; + +TEST_F(DynCommonTests, ParseValidName) { + stream = fmemopen((void*)"valid_name", 10, "r"); + ASSERT_EQ(dynCommon_parseName(stream, &result), 0); + ASSERT_STREQ(result, "valid_name"); +} + +TEST_F(DynCommonTests, ParseNameTillInvalidCharacter) { + stream = fmemopen((void*)"invalid-name", 12, "r"); + ASSERT_EQ(dynCommon_parseName(stream, &result), 0); + ASSERT_STREQ("invalid", result); + ASSERT_EQ('-', fgetc(stream)); +} + +TEST_F(DynCommonTests, ParseNameWithExtraAllowableCharacters) { + stream = fmemopen((void*)"invalid-@name", 13, "r"); + ASSERT_EQ(dynCommon_parseNameAlsoAccept(stream, "@-", &result), 0); + ASSERT_STREQ("invalid-@name", result); + ASSERT_EQ(EOF, fgetc(stream)); + +} + +TEST_F(DynCommonTests, ParseEmptyName) { + stream = fmemopen((void*)"", 1, "r"); + ASSERT_EQ(dynCommon_parseName(stream, &result), 1); + ASSERT_EQ('\0', fgetc(stream)); +} + +TEST_F(DynCommonTests, EatCharValid) { + stream = fmemopen((void*)"valid", 5, "r"); + ASSERT_EQ(dynCommon_eatChar(stream, 'v'), 0); + ASSERT_EQ(dynCommon_eatChar(stream, 'a'), 0); + ASSERT_EQ(dynCommon_eatChar(stream, 'l'), 0); + ASSERT_EQ(dynCommon_eatChar(stream, 'i'), 0); + ASSERT_EQ(dynCommon_eatChar(stream, 'd'), 0); + ASSERT_FALSE(feof(stream)); + ASSERT_EQ(dynCommon_eatChar(stream, EOF), 0); + ASSERT_TRUE(feof(stream)); +} + +TEST_F(DynCommonTests, EatCharFromEmptyString) { + stream = fmemopen((void*)"", 1, "r"); + ASSERT_EQ(dynCommon_eatChar(stream, '\0'), 0); + ASSERT_FALSE(feof(stream)); + ASSERT_EQ(dynCommon_eatChar(stream, EOF), 0); + ASSERT_TRUE(feof(stream)); +} + +TEST_F(DynCommonTests, ParseNameValueSection) { + stream = fmemopen((void*)"name1=value1\nname2=value2\n", 26, "r"); + struct namvals_head head; + TAILQ_INIT(&head); + ASSERT_EQ(dynCommon_parseNameValueSection(stream, &head), 0); + + struct namval_entry *entry = TAILQ_FIRST(&head); + ASSERT_STREQ("name1", entry->name); + ASSERT_STREQ("value1", entry->value); + + entry = TAILQ_NEXT(entry, entries); + ASSERT_STREQ("name2", entry->name); + ASSERT_STREQ("value2", entry->value); + + dynCommon_clearNamValHead(&head); +} + +TEST_F(DynCommonTests, ParseNameValueSectionWithPairWithEmptyName) { + stream = fmemopen((void*)"=value1\nname2=value2\n", 22, "r"); + struct namvals_head head; + TAILQ_INIT(&head); + ASSERT_EQ(dynCommon_parseNameValueSection(stream, &head), 1); + ASSERT_STREQ("Parsed empty name", celix_err_popLastError()); + + dynCommon_clearNamValHead(&head); +} + +TEST_F(DynCommonTests, ParseNameValueSectionWithPairWithEmptyValue) { + stream = fmemopen((void*)"name1=\nname2=value2\n", 22, "r"); + struct namvals_head head; + TAILQ_INIT(&head); + ASSERT_EQ(dynCommon_parseNameValueSection(stream, &head), 1); + ASSERT_STREQ("Parsed empty name", celix_err_popLastError()); + + dynCommon_clearNamValHead(&head); +} + +TEST_F(DynCommonTests, ParseNameValueSectionWithPairMissingEquality) { + stream = fmemopen((void*)"name1 value1\nname2=value2\n", 22, "r"); + struct namvals_head head; + TAILQ_INIT(&head); + ASSERT_EQ(dynCommon_parseNameValueSection(stream, &head), 1); + ASSERT_STREQ("Error parsing, expected token '=' got ' ' at position 6", celix_err_popLastError()); + + dynCommon_clearNamValHead(&head); +} + +TEST_F(DynCommonTests, ParseNameValueSectionWithPairMissingNewline) { + stream = fmemopen((void*)"name1=value1 name2=value2\n", 26, "r"); + struct namvals_head head; + TAILQ_INIT(&head); + ASSERT_EQ(dynCommon_parseNameValueSection(stream, &head), 1); + ASSERT_STREQ("Error parsing, expected token '\n' got ' ' at position 13", celix_err_popLastError()); + + dynCommon_clearNamValHead(&head); +} \ No newline at end of file diff --git a/libs/dfi/gtest/src/dyn_example_functions.c b/libs/dfi/gtest/src/dyn_example_functions.c index 091382a18..3f95c482c 100644 --- a/libs/dfi/gtest/src/dyn_example_functions.c +++ b/libs/dfi/gtest/src/dyn_example_functions.c @@ -46,3 +46,7 @@ void example5Func(const char *s1, char *s2) { assert(strncmp("s1", s1, 5) == 0); assert(strncmp("s2", s2, 5) == 0); } + +int32_t example6Func() { + return 1234; +} \ No newline at end of file diff --git a/libs/dfi/gtest/src/dyn_example_functions.h b/libs/dfi/gtest/src/dyn_example_functions.h index 41700467b..8d50e4e30 100644 --- a/libs/dfi/gtest/src/dyn_example_functions.h +++ b/libs/dfi/gtest/src/dyn_example_functions.h @@ -50,6 +50,8 @@ void example4Func(struct tst_seq seq); #define EXAMPLE5_DESCRIPTOR "example(#const=true;tt)V" void example5Func(const char *s1, char *s2); +#define EXAMPLE6_DESCRIPTOR "example()I" +int32_t example6Func(); #ifdef __cplusplus diff --git a/libs/dfi/gtest/src/dyn_function_ei_tests.cc b/libs/dfi/gtest/src/dyn_function_ei_tests.cc new file mode 100644 index 000000000..86d5df180 --- /dev/null +++ b/libs/dfi/gtest/src/dyn_function_ei_tests.cc @@ -0,0 +1,112 @@ +/* + * 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. + */ + +#include "dyn_function.h" +#include "dyn_example_functions.h" + +#include "celix_err.h" +#include "asprintf_ei.h" +#include "ffi_ei.h" +#include "malloc_ei.h" +#include "stdio_ei.h" + +#include +#include +#include +#include + +class DynFunctionErrorInjectionTestSuite : public ::testing::Test { +public: + DynFunctionErrorInjectionTestSuite() = default; + ~DynFunctionErrorInjectionTestSuite() override { + celix_ei_expect_asprintf(nullptr, 0, -1); + celix_ei_expect_fmemopen(nullptr, 0, nullptr); + celix_ei_expect_ffi_prep_cif(nullptr, 0, FFI_OK); + celix_ei_expect_calloc(nullptr, 0, nullptr); + celix_err_resetErrors(); + } +}; + +TEST_F(DynFunctionErrorInjectionTestSuite, ParseError) { + dyn_function_type *dynFunc = nullptr; + int rc; + celix_ei_expect_calloc((void*)dynFunction_parse, 0, nullptr); + rc = dynFunction_parseWithStr(EXAMPLE1_DESCRIPTOR, nullptr, &dynFunc); + EXPECT_NE(0, rc); + EXPECT_STREQ("Error allocating memory for dyn function", celix_err_popLastError()); + + celix_ei_expect_ffi_prep_cif((void*)dynFunction_parse, 1, FFI_BAD_TYPEDEF); + rc = dynFunction_parseWithStr(EXAMPLE1_DESCRIPTOR, nullptr, &dynFunc); + EXPECT_NE(0, rc); + EXPECT_STREQ("Error ffi_prep_cif 1", celix_err_popLastError()); + + celix_ei_expect_calloc((void*)dynFunction_parse, 1, nullptr, 4); + rc = dynFunction_parseWithStr(EXAMPLE1_DESCRIPTOR, nullptr, &dynFunc); + EXPECT_NE(0, rc); + EXPECT_STREQ("Error allocating memory for ffi args", celix_err_popLastError()); + + celix_ei_expect_fmemopen((void*)dynFunction_parseWithStr, 0, nullptr); + rc = dynFunction_parseWithStr(EXAMPLE1_DESCRIPTOR, nullptr, &dynFunc); + EXPECT_NE(0, rc); + std::string result = "Error creating mem stream for descriptor string. "; + result += strerror(ENOMEM); + EXPECT_STREQ(result.c_str(), celix_err_popLastError()); + + celix_ei_expect_calloc((void*)dynFunction_parse, 1, nullptr, 3); + rc = dynFunction_parseWithStr(EXAMPLE1_DESCRIPTOR, nullptr, &dynFunc); + EXPECT_NE(0, rc); + EXPECT_STREQ("Error parsing descriptor", celix_err_popLastError()); + EXPECT_STREQ("Error allocating arg", celix_err_popLastError()); +} + +class DynClosureErrorInjectionTestSuite : public ::testing::Test { +public: + DynClosureErrorInjectionTestSuite() = default; + ~DynClosureErrorInjectionTestSuite() override { + celix_ei_expect_ffi_prep_closure_loc(nullptr, 0, FFI_OK); + celix_ei_expect_ffi_closure_alloc(nullptr, 0, nullptr); + } +}; + + +static void example1_binding(void*, void* args[], void *out) { + int32_t a = *((int32_t *)args[0]); + int32_t b = *((int32_t *)args[1]); + int32_t c = *((int32_t *)args[2]); + int32_t *ret = (int32_t *)out; + *ret = a + b + c; +} + +TEST_F(DynClosureErrorInjectionTestSuite, CreatError) { + celix_autoptr(dyn_function_type) dynFunc = nullptr; + int rc; + rc = dynFunction_parseWithStr(EXAMPLE1_DESCRIPTOR, nullptr, &dynFunc); + ASSERT_EQ(0, rc); + + int32_t (*func)(int32_t a, int32_t b, int32_t c) = NULL; + celix_ei_expect_ffi_closure_alloc((void*)dynFunction_createClosure, 0, nullptr); + rc = dynFunction_createClosure(dynFunc, example1_binding, NULL, (void(**)(void))&func); + ASSERT_NE(0, rc); + EXPECT_NE(0, dynFunction_getFnPointer(dynFunc, (void(**)(void))&func)); + + celix_ei_expect_ffi_prep_closure_loc((void*)dynFunction_createClosure, 0, FFI_BAD_ABI); + rc = dynFunction_createClosure(dynFunc, example1_binding, NULL, (void(**)(void))&func); + ASSERT_NE(0, rc); + EXPECT_NE(0, dynFunction_getFnPointer(dynFunc, (void(**)(void))&func)); +} \ No newline at end of file diff --git a/libs/dfi/gtest/src/dyn_function_tests.cpp b/libs/dfi/gtest/src/dyn_function_tests.cpp index 7f0ba0dc8..d4e302469 100644 --- a/libs/dfi/gtest/src/dyn_function_tests.cpp +++ b/libs/dfi/gtest/src/dyn_function_tests.cpp @@ -17,25 +17,14 @@ * under the License. */ -#include "gtest/gtest.h" -#include #include "dyn_example_functions.h" - #include "dyn_common.h" #include "dyn_function.h" -#include - -static void stdLog(void*, int level, const char *file, int line, const char *msg, ...) { - va_list ap; - const char *levels[5] = {"NIL", "ERROR", "WARNING", "INFO", "DEBUG"}; - fprintf(stderr, "%s: FILE:%s, LINE:%i, MSG:",levels[level], file, line); - va_start(ap, msg); - vfprintf(stderr, msg, ap); - fprintf(stderr, "\n"); - va_end(ap); -} +#include "celix_err.h" +#include +#include #define INVALID_FUNC_DESCRIPTOR "example$[D)V"//$ is an invalid symbol, missing ( #define INVALID_FUNC_TYPE_DESCRIPTOR "example(H)A"//H and A are invalid types @@ -44,46 +33,44 @@ static void stdLog(void*, int level, const char *file, int line, const char *msg class DynFunctionTests : public ::testing::Test { public: DynFunctionTests() { - int lvl = 1; - dynFunction_logSetup(stdLog, nullptr, lvl); - dynType_logSetup(stdLog, nullptr, lvl); - dynCommon_logSetup(stdLog, nullptr, lvl); } - ~DynFunctionTests() override = default; + ~DynFunctionTests() override { + celix_err_resetErrors(); + } }; +TEST_F(DynFunctionTests, DynFuncTest1) { + dyn_function_type *dynFunc = nullptr; + int rc; + void (*fp)(void) = (void (*)(void)) example1; + + rc = dynFunction_parseWithStr(EXAMPLE1_DESCRIPTOR, nullptr, &dynFunc); + ASSERT_EQ(0, rc); + EXPECT_TRUE(dynFunction_hasReturn(dynFunc)); + EXPECT_EQ(3, dynFunction_nrOfArguments(dynFunc)); + EXPECT_STREQ("example", dynFunction_getName(dynFunc)); + auto args = dynFunction_arguments(dynFunc); + dyn_function_argument_type* arg = NULL; + TAILQ_FOREACH(arg, args, entries) { + EXPECT_EQ(DYN_FUNCTION_ARGUMENT_META__STD, arg->argumentMeta); + EXPECT_EQ('I', dynType_descriptorType(arg->type)); + } -extern "C" { - static bool func_test1() { - dyn_function_type *dynFunc = nullptr; - int rc; - void (*fp)(void) = (void (*)(void)) example1; - - rc = dynFunction_parseWithStr(EXAMPLE1_DESCRIPTOR, nullptr, &dynFunc); - - ffi_sarg rVal = 0; - int32_t a = 2; - int32_t b = 4; - int32_t c = 8; - void *values[3]; - values[0] = &a; - values[1] = &b; - values[2] = &c; - - if (rc == 0) { - rc = dynFunction_call(dynFunc, fp, &rVal, values); - dynFunction_destroy(dynFunc); - } + ffi_sarg rVal = 0; + int32_t a = 2; + int32_t b = 4; + int32_t c = 8; + void *values[3]; + values[0] = &a; + values[1] = &b; + values[2] = &c; - return rc == 0 && rVal == 14; - } -} + rc = dynFunction_call(dynFunc, fp, &rVal, values); + dynFunction_destroy(dynFunc); -TEST_F(DynFunctionTests, DynFuncTest1) { - //NOTE only using libffi with extern C, because combining libffi with EXPECT_*/ASSERT_* call leads to - //corrupted memory. Note that libffi is a function for interfacing with C not C++ - EXPECT_TRUE(func_test1()); + EXPECT_EQ(0, rc); + EXPECT_EQ(14, rVal); } extern "C" { @@ -130,15 +117,15 @@ static bool func_acc() { int nrOfArgs = 0; bool isStruct = false; bool isVoid = false; - dyn_type *nonExist = nullptr; + const dyn_type *nonExist = nullptr; if (rc == 0) { nrOfArgs = dynFunction_nrOfArguments(dynFunc); - dyn_type *arg1 = dynFunction_argumentTypeForIndex(dynFunc, 1); + const dyn_type *arg1 = dynFunction_argumentTypeForIndex(dynFunc, 1); if (arg1 != nullptr) { isStruct = '{' == dynType_descriptorType(arg1); } nonExist = dynFunction_argumentTypeForIndex(dynFunc, 10); - dyn_type *returnType = dynFunction_returnType(dynFunc); + const dyn_type* returnType = dynFunction_returnType(dynFunc); if (returnType != nullptr) { isVoid = 'V' == dynType_descriptorType(returnType); } @@ -203,13 +190,15 @@ TEST_F(DynFunctionTests, DynFuncTest3) { EXPECT_TRUE(func_test3()); } -extern "C" { -static bool func_test4() { +TEST_F(DynFunctionTests, DynFuncTest4) { dyn_function_type *dynFunc = nullptr; void (*fp)(void) = (void(*)(void)) example4Func; int rc; rc = dynFunction_parseWithStr(EXAMPLE4_DESCRIPTOR, nullptr, &dynFunc); + ASSERT_EQ(0, rc); + EXPECT_FALSE(dynFunction_hasReturn(dynFunc)); + EXPECT_EQ(1, dynFunction_nrOfArguments(dynFunc)); double buf[4]; buf[0] = 1.1; @@ -221,19 +210,9 @@ static bool func_test4() { void *args[1]; args[0] = &seq; - if (rc == 0) { - rc = dynFunction_call(dynFunc, fp, nullptr, args); - dynFunction_destroy(dynFunc); - } - - return rc == 0; -} -} - -TEST_F(DynFunctionTests, DynFuncTest4) { - //NOTE only using libffi with extern C, because combining libffi with EXPECT_*/ASSERT_* call leads to - //corrupted memory. Note that libffi is a function for interfacing with C not C++ - EXPECT_TRUE(func_test4()); + rc = dynFunction_call(dynFunc, fp, nullptr, args); + dynFunction_destroy(dynFunc); + EXPECT_EQ(0, rc); } extern "C" { @@ -267,20 +246,77 @@ TEST_F(DynFunctionTests, DynFuncTest5) { EXPECT_TRUE(func_test5()); } -extern "C" { -static bool func_invalid() { +TEST_F(DynFunctionTests, DynFuncTest6) { + dyn_function_type *dynFunc = nullptr; + void (*fp)(void) = (void(*)(void)) example6Func; + int rc; + + rc = dynFunction_parseWithStr(EXAMPLE6_DESCRIPTOR, nullptr, &dynFunc); + ASSERT_EQ(0, rc); + EXPECT_TRUE(dynFunction_hasReturn(dynFunc)); + EXPECT_EQ(0, dynFunction_nrOfArguments(dynFunc)); + auto args = dynFunction_arguments(dynFunc); + EXPECT_TRUE(TAILQ_EMPTY(args)); + + ffi_sarg rVal = 0; + rc = dynFunction_call(dynFunc, fp, &rVal, nullptr); + dynFunction_destroy(dynFunc); + EXPECT_EQ(0, rc); + EXPECT_EQ(1234, rVal); +} + +TEST_F(DynFunctionTests, InvalidDynFuncTest) { + dyn_function_type *dynFunc = nullptr; int rc1 = dynFunction_parseWithStr(INVALID_FUNC_DESCRIPTOR, nullptr, &dynFunc); + EXPECT_NE(0, rc1); + EXPECT_STREQ("Error parsing descriptor", celix_err_popLastError()); + EXPECT_STREQ("Error parsing, expected token '(' got '$' at position 8", celix_err_popLastError()); dynFunc = nullptr; int rc2 = dynFunction_parseWithStr(INVALID_FUNC_TYPE_DESCRIPTOR, nullptr, &dynFunc); - return rc1 != 0 && rc2 != 0; -} -} + EXPECT_NE(0, rc2); + EXPECT_STREQ("Error parsing descriptor", celix_err_popLastError()); + EXPECT_STREQ("Error unsupported type 'H'", celix_err_popLastError()); -TEST_F(DynFunctionTests, InvalidDynFuncTest) { - //NOTE only using libffi with extern C, because combining libffi with EXPECT_*/ASSERT_* call leads to - //corrupted memory. Note that libffi is a function for interfacing with C not C++ - EXPECT_TRUE(func_invalid()); + dynFunc = nullptr; + int rc3 = dynFunction_parseWithStr("$xample(III)I", nullptr, &dynFunc); + EXPECT_NE(0, rc3); + EXPECT_STREQ("Error parsing descriptor", celix_err_popLastError()); + EXPECT_STREQ("Parsed empty name", celix_err_popLastError()); + + dynFunc = nullptr; + int rc4 = dynFunction_parseWithStr("example(III", nullptr, &dynFunc); + EXPECT_NE(0, rc4); + EXPECT_STREQ("Error parsing descriptor", celix_err_popLastError()); + EXPECT_STREQ("Error missing ')'", celix_err_popLastError()); } +TEST_F(DynFunctionTests, WrongArgumentMetaTest) { + dyn_function_type *dynFunc = nullptr; + int rc1 = dynFunction_parseWithStr("example(#am=handle;tt)N", nullptr, &dynFunc); + EXPECT_NE(0, rc1); + EXPECT_STREQ("Error 'handle' is only allowed for untyped pointer not 't'", celix_err_popLastError()); + + int rc2 = dynFunction_parseWithStr("example(#am=handle;THandle=P;lHandle;t)N", nullptr, &dynFunc); + EXPECT_EQ(0, rc2); + dynFunction_destroy(dynFunc); + + int rc3 = dynFunction_parseWithStr("example(#am=handle;P#am=out;t)N", nullptr, &dynFunc); + EXPECT_NE(0, rc3); + EXPECT_STREQ("Error 'out' is only allowed for typed pointer not 't'", celix_err_popLastError()); + + int rc4 = dynFunction_parseWithStr("example(#am=handle;P#am=out;*D)N", nullptr, &dynFunc); + EXPECT_NE(0, rc4); + EXPECT_STREQ("Error 'out' is only allowed for pointer to text or typed pointer not to 'D'", celix_err_popLastError()); + + // #am=pre argument is not allowed for non pointer types + int rc5 = dynFunction_parseWithStr("example(#am=pre;I)N", nullptr, &dynFunc); + EXPECT_NE(0, rc5); + EXPECT_STREQ("Error 'pre' is only allowed for typed pointer not 'I'", celix_err_popLastError()); + + // #am=pre argument is not allowed for pointer to nontrivial types + int rc6 = dynFunction_parseWithStr("example(#am=pre;**D)N", nullptr, &dynFunc); + EXPECT_NE(0, rc6); + EXPECT_STREQ("Error 'pre' is only allowed for pointer to trivial types not non-trivial '*'", celix_err_popLastError()); +} diff --git a/libs/dfi/gtest/src/dyn_interface_ei_tests.cc b/libs/dfi/gtest/src/dyn_interface_ei_tests.cc new file mode 100644 index 000000000..cfa1931bc --- /dev/null +++ b/libs/dfi/gtest/src/dyn_interface_ei_tests.cc @@ -0,0 +1,101 @@ +/* + * 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. + */ + +#include "dyn_interface.h" +#include "dyn_descriptor.h" +#include "dyn_common.h" +#include "celix_err.h" +#include "celix_stdio_cleanup.h" +#include "malloc_ei.h" +#include "stdio_ei.h" + +#include +#include +#include +#include + +class DynInterfaceErrorInjectionTestSuite : public ::testing::Test { +public: + DynInterfaceErrorInjectionTestSuite() = default; + + ~DynInterfaceErrorInjectionTestSuite() override { + celix_ei_expect_fgetc(nullptr, 0, EOF); + celix_ei_expect_calloc(nullptr, 0, nullptr); + celix_ei_expect_open_memstream(nullptr, 0, nullptr); + } +}; + +TEST_F(DynInterfaceErrorInjectionTestSuite, ParseError) { + int status = 0; + dyn_interface_type *dynIntf = NULL; + celix_autoptr(FILE) desc = fopen("descriptors/example1.descriptor", "r"); + ASSERT_NE(nullptr, desc); + + // not enough memory for dyn_interface_type + celix_ei_expect_calloc((void*) dynInterface_parse, 0, nullptr); + status = dynInterface_parse(desc, &dynIntf); + ASSERT_NE(0, status); + ASSERT_STREQ("Error allocating memory for dynamic interface", celix_err_popLastError()); + + rewind(desc); + // not enough memory for namval_entry when parsing header section + celix_ei_expect_calloc((void*) dynInterface_parse, 3, nullptr, 1); + status = dynInterface_parse(desc, &dynIntf); + ASSERT_NE(0, status); + ASSERT_STREQ("Error allocating memory for namval entry", celix_err_popLastError()); + + rewind(desc); + // not enough memory for type_entry when parsing types section + celix_ei_expect_calloc((void*) celix_dynDescriptor_parse, 2, nullptr, 5); + status = dynInterface_parse(desc, &dynIntf); + ASSERT_NE(0, status); + ASSERT_STREQ("Error allocating memory for type entry", celix_err_popLastError()); + + rewind(desc); + // not enough memory for method_entry when parsing methods section + celix_ei_expect_calloc((void*) celix_dynDescriptor_parse, 3, nullptr, 1); + status = dynInterface_parse(desc, &dynIntf); + ASSERT_NE(0, status); + ASSERT_STREQ("Error allocating memory for method entry", celix_err_popLastError()); + + rewind(desc); + // not enough memory for open_memstream + celix_ei_expect_open_memstream((void*) dynCommon_parseNameAlsoAccept, 0, nullptr); + status = dynInterface_parse(desc, &dynIntf); + ASSERT_NE(0, status); + std::string msg = "Error creating mem stream for name. "; + msg += strerror(ENOMEM); + ASSERT_STREQ(msg.c_str(), celix_err_popLastError()); + + rewind(desc); + // encounter EOF when parsing header section + celix_ei_expect_fgetc((void*) dynCommon_eatChar, 0, EOF); + status = dynInterface_parse(desc, &dynIntf); + ASSERT_NE(0, status); + msg = "Error parsing, expected token"; + ASSERT_TRUE(msg.compare(0, msg.length(), celix_err_popLastError())); + + rewind(desc); + // encounter EOF when expecting newline ending a header name + celix_ei_expect_fgetc((void*) dynCommon_eatChar, 0, EOF, 2); + status = dynInterface_parse(desc, &dynIntf); + ASSERT_NE(0, status); + msg = "Error parsing, expected token"; + ASSERT_TRUE(msg.compare(0, msg.length(), celix_err_popLastError())); +} \ No newline at end of file diff --git a/libs/dfi/gtest/src/dyn_interface_tests.cpp b/libs/dfi/gtest/src/dyn_interface_tests.cpp index 2ddaa1473..58443521a 100644 --- a/libs/dfi/gtest/src/dyn_interface_tests.cpp +++ b/libs/dfi/gtest/src/dyn_interface_tests.cpp @@ -19,8 +19,6 @@ #include "gtest/gtest.h" -#include - extern "C" { @@ -33,38 +31,19 @@ extern "C" { #include "dyn_common.h" #include "dyn_interface.h" -#include "version.h" - -#if NO_MEMSTREAM_AVAILABLE -#include "open_memstream.h" -#include "fmemopen.h" -#endif - - static void stdLog(void*, int level, const char *file, int line, const char *msg, ...) { - va_list ap; - const char *levels[5] = {"NIL", "ERROR", "WARNING", "INFO", "DEBUG"}; - fprintf(stderr, "%s: FILE:%s, LINE:%i, MSG:",levels[level], file, line); - va_start(ap, msg); - vfprintf(stderr, msg, ap); - fprintf(stderr, "\n"); - va_end(ap); - } +#include "celix_err.h" +#include "celix_version.h" static void checkInterfaceVersion(dyn_interface_type* dynIntf, const char* v) { - int status; - - char *version = NULL; - status = dynInterface_getVersionString(dynIntf, &version); - ASSERT_EQ(0, status); + const char* version = dynInterface_getVersionString(dynIntf); ASSERT_STREQ(v, version); - version_pt msgVersion = NULL, localMsgVersion = NULL; + const celix_version_t* msgVersion = dynInterface_getVersion(dynIntf); + celix_version_t* localMsgVersion = NULL; int cmpVersion = -1; - version_createVersionFromString(version, &localMsgVersion); - status = dynInterface_getVersion(dynIntf, &msgVersion); - ASSERT_EQ(0, status); - version_compareTo(msgVersion, localMsgVersion, &cmpVersion); + localMsgVersion = celix_version_createVersionFromString(version); + cmpVersion = celix_version_compareTo(msgVersion, localMsgVersion); ASSERT_EQ(cmpVersion, 0); - version_destroy(localMsgVersion); + celix_version_destroy(localMsgVersion); } static void test1(void) { @@ -76,26 +55,29 @@ extern "C" { ASSERT_EQ(0, status); fclose(desc); - char *name = NULL; - status = dynInterface_getName(dynIntf, &name); - ASSERT_EQ(0, status); + const char *name = dynInterface_getName(dynIntf); ASSERT_STREQ("calculator", name); - checkInterfaceVersion(dynIntf,"1.0.0"); + checkInterfaceVersion(dynIntf,"1.0.0"); - char *annVal = NULL; + auto method = dynInterface_findMethod(dynIntf, "add(DD)D"); + EXPECT_EQ(DYN_FUNCTION_ARGUMENT_META__HANDLE, dynFunction_argumentMetaForIndex(method->dynFunc, 0)); + EXPECT_EQ(DYN_FUNCTION_ARGUMENT_META__STD, dynFunction_argumentMetaForIndex(method->dynFunc, 1)); + EXPECT_EQ(DYN_FUNCTION_ARGUMENT_META__STD, dynFunction_argumentMetaForIndex(method->dynFunc, 2)); + EXPECT_EQ(DYN_FUNCTION_ARGUMENT_META__PRE_ALLOCATED_OUTPUT, dynFunction_argumentMetaForIndex(method->dynFunc, 3)); + EXPECT_EQ(DYN_FUNCTION_ARGUMENT_META__STD, dynFunction_argumentMetaForIndex(method->dynFunc, 4)); + + const char *annVal = NULL; status = dynInterface_getAnnotationEntry(dynIntf, "classname", &annVal); ASSERT_EQ(0, status); ASSERT_STREQ("org.example.Calculator", annVal); - char *nonExist = NULL; + const char *nonExist = NULL; status = dynInterface_getHeaderEntry(dynIntf, "nonExisting", &nonExist); ASSERT_TRUE(status != 0); ASSERT_TRUE(nonExist == NULL); - struct methods_head *list = NULL; - status = dynInterface_methods(dynIntf, &list); - ASSERT_TRUE(status == 0); + const struct methods_head* list = dynInterface_methods(dynIntf); ASSERT_TRUE(list != NULL); int count = dynInterface_nrOfMethods(dynIntf); @@ -124,9 +106,9 @@ extern "C" { FILE *desc = fopen("descriptors/invalids/invalid.descriptor", "r"); assert(desc != NULL); status = dynInterface_parse(desc, &dynIntf); - //dynInterface_destroy(dynIntf); ASSERT_EQ(1, status); //Test fails because of a space at the end of the name fclose(desc); desc=NULL; + celix_err_resetErrors(); /* Invalid header */ desc = fopen("descriptors/invalids/invalidHeader.descriptor", "r"); @@ -134,46 +116,175 @@ extern "C" { status = dynInterface_parse(desc, &dynIntf); ASSERT_EQ(1, status); //Test fails because of missing name value fclose(desc); desc=NULL; + celix_err_resetErrors(); /* Header without Version */ desc = fopen("descriptors/invalids/noVersion.descriptor", "r"); assert(desc != NULL); status = dynInterface_parse(desc, &dynIntf); - //dynInterface_destroy(dynIntf); ASSERT_EQ(1, status); //Test fails because of missing version field in header section fclose(desc); desc=NULL; + celix_err_resetErrors(); + + /* Header without Type */ + desc = fopen("descriptors/invalids/noType.descriptor", "r"); + assert(desc != NULL); + status = dynInterface_parse(desc, &dynIntf); + ASSERT_EQ(1, status); //Test fails because of missing type field in header section + fclose(desc); desc=NULL; + celix_err_resetErrors(); + + /* Header without Name */ + desc = fopen("descriptors/invalids/noName.descriptor", "r"); + assert(desc != NULL); + status = dynInterface_parse(desc, &dynIntf); + ASSERT_EQ(1, status); //Test fails because of missing name field in header section + fclose(desc); desc=NULL; + celix_err_resetErrors(); + + /* Invalid Annotations Section */ + desc = fopen("descriptors/invalids/invalidInterfaceAnnotations.descriptor", "r"); + assert(desc != NULL); + status = dynInterface_parse(desc, &dynIntf); + ASSERT_EQ(1, status); //Test fails because of missing name field in annotations section + fclose(desc); desc=NULL; + celix_err_resetErrors(); /* Invalid section */ desc = fopen("descriptors/invalids/invalidSection.descriptor", "r"); assert(desc != NULL); status = dynInterface_parse(desc, &dynIntf); - //dynInterface_destroy(dynIntf); ASSERT_EQ(1, status); //Test fails because of unknown section type fclose(desc); desc=NULL; + celix_err_resetErrors(); /* Invalid return type */ desc = fopen("descriptors/invalids/invalidMethodReturnType.descriptor", "r"); assert(desc != NULL); status = dynInterface_parse(desc, &dynIntf); - //dynInterface_destroy(dynIntf); ASSERT_EQ(1, status); //Test fails because of invalid return type (D instead of N) + EXPECT_STREQ("Parse Error. Got return type 'D' rather than 'N' (native int) for method add(DD)D (0)", celix_err_popLastError()); + fclose(desc); desc=NULL; + celix_err_resetErrors(); + + /* Method without arguments */ + desc = fopen("descriptors/invalids/methodWithoutArguments.descriptor", "r"); + assert(desc != NULL); + status = dynInterface_parse(desc, &dynIntf); + ASSERT_NE(0, status); + EXPECT_STREQ("Parse Error. The first argument must be handle for method add(DD)D (0)", celix_err_popLastError()); + fclose(desc); desc=NULL; + celix_err_resetErrors(); + + /* Method missing handle */ + desc = fopen("descriptors/invalids/methodMissingHandle.descriptor", "r"); + assert(desc != NULL); + status = dynInterface_parse(desc, &dynIntf); + ASSERT_NE(0, status); + EXPECT_STREQ("Parse Error. The first argument must be handle for method add(DD)D (0)", celix_err_popLastError()); + fclose(desc); desc=NULL; + celix_err_resetErrors(); + + /* Method with multiple handles */ + desc = fopen("descriptors/invalids/methodWithMultipleHandles.descriptor", "r"); + assert(desc != NULL); + status = dynInterface_parse(desc, &dynIntf); + ASSERT_NE(0, status); + EXPECT_STREQ("Parse Error. Handle argument is only allowed as the first argument for method add(DD)D (0)", celix_err_popLastError()); + fclose(desc); desc=NULL; + celix_err_resetErrors(); + + /* Method with multiple PreOut arguments */ + desc = fopen("descriptors/invalids/multiPreOutArgs.descriptor", "r"); + assert(desc != NULL); + status = dynInterface_parse(desc, &dynIntf); + ASSERT_NE(0, status); + EXPECT_STREQ("Parse Error. Output argument is only allowed as the last argument for method multiPreOut(V)Di (0)", celix_err_popLastError()); + fclose(desc); desc=NULL; + celix_err_resetErrors(); + + /* Method with multiple Out arguments */ + desc = fopen("descriptors/invalids/multiOutArgs.descriptor", "r"); + assert(desc != NULL); + status = dynInterface_parse(desc, &dynIntf); + ASSERT_NE(0, status); + EXPECT_STREQ("Parse Error. Output argument is only allowed as the last argument for method multiOut(V)Di (0)", celix_err_popLastError()); + fclose(desc); desc=NULL; + celix_err_resetErrors(); + + /* Method with an Out argument at wrong position */ + desc = fopen("descriptors/invalids/outArgAtWrongPosition.descriptor", "r"); + assert(desc != NULL); + status = dynInterface_parse(desc, &dynIntf); + ASSERT_NE(0, status); + EXPECT_STREQ("Parse Error. Output argument is only allowed as the last argument for method outWrongPos(V)Di (0)", + celix_err_popLastError()); fclose(desc); desc=NULL; + celix_err_resetErrors(); /* Invalid method section */ desc = fopen("descriptors/invalids/invalidMethod.descriptor", "r"); assert(desc != NULL); status = dynInterface_parse(desc, &dynIntf); - //dynInterface_destroy(dynIntf); ASSERT_EQ(1, status); //Test fails because of space at the end of the method fclose(desc); desc=NULL; + celix_err_resetErrors(); + + /* Invalid method section missing method id */ + desc = fopen("descriptors/invalids/invalidMethodMissingId.descriptor", "r"); + assert(desc != NULL); + status = dynInterface_parse(desc, &dynIntf); + ASSERT_EQ(1, status); + fclose(desc); desc=NULL; + celix_err_resetErrors(); + + /* Invalid method section missing equality */ + desc = fopen("descriptors/invalids/invalidMethodMissingEquality.descriptor", "r"); + assert(desc != NULL); + status = dynInterface_parse(desc, &dynIntf); + ASSERT_EQ(1, status); + fclose(desc); desc=NULL; + celix_err_resetErrors(); + + /* Invalid method section missing function name */ + desc = fopen("descriptors/invalids/invalidMethodMissingFunctionName.descriptor", "r"); + assert(desc != NULL); + status = dynInterface_parse(desc, &dynIntf); + ASSERT_EQ(1, status); + fclose(desc); desc=NULL; + celix_err_resetErrors(); /* Invalid type */ desc = fopen("descriptors/invalids/invalidType.descriptor", "r"); assert(desc != NULL); status = dynInterface_parse(desc, &dynIntf); - //dynInterface_destroy(dynIntf); ASSERT_EQ(1, status); //Test fails because of space at the end of the type fclose(desc); desc=NULL; + celix_err_resetErrors(); + + /* Invalid types section missing type name */ + desc = fopen("descriptors/invalids/noTypeName.descriptor", "r"); + assert(desc != NULL); + status = dynInterface_parse(desc, &dynIntf); + ASSERT_EQ(1, status); //Test fails because of missing type name + fclose(desc); desc=NULL; + celix_err_resetErrors(); + + /* Invalid types section missing equality */ + desc = fopen("descriptors/invalids/invalidTypeMissingEquality.descriptor", "r"); + assert(desc != NULL); + status = dynInterface_parse(desc, &dynIntf); + ASSERT_EQ(1, status); //Test fails because of missing equality + fclose(desc); desc=NULL; + celix_err_resetErrors(); + + /* Invalid types section with unrecognized simple type */ + desc = fopen("descriptors/invalids/invalidTypeUnrecognizedSimpleType.descriptor", "r"); + assert(desc != NULL); + status = dynInterface_parse(desc, &dynIntf); + ASSERT_NE(0, status); + fclose(desc); desc=NULL; + celix_err_resetErrors(); /* Invalid metatype in method description */ desc = fopen("descriptors/invalids/invalidMetaType.descriptor", "r"); @@ -182,28 +293,40 @@ extern "C" { dynInterface_destroy(dynIntf); ASSERT_EQ(0, status); //Invalid meta type doesn't generate errors, just warnings fclose(desc); desc=NULL; dynIntf=NULL; + celix_err_resetErrors(); /* Invalid version section */ desc = fopen("descriptors/invalids/invalidVersion.descriptor", "r"); assert(desc != NULL); status = dynInterface_parse(desc, &dynIntf); - //dynInterface_destroy(dynIntf); - ASSERT_EQ(1, status); //Invalid meta type doesn't generate errors, just warnings + ASSERT_EQ(1, status); fclose(desc); desc=NULL; + celix_err_resetErrors(); + /* garbage descriptor */ + desc = fopen("descriptors/invalids/garbage.descriptor", "r"); + assert(desc != NULL); + status = dynInterface_parse(desc, &dynIntf); + ASSERT_EQ(1, status); + fclose(desc); desc=NULL; + celix_err_resetErrors(); + + /* invalid extra section */ + desc = fopen("descriptors/invalids/invalidExtraSection.descriptor", "r"); + assert(desc != NULL); + status = dynInterface_parse(desc, &dynIntf); + ASSERT_EQ(1, status); + fclose(desc); desc=NULL; + celix_err_resetErrors(); } } class DynInterfaceTests : public ::testing::Test { public: DynInterfaceTests() { - int level = 1; - dynCommon_logSetup(stdLog, NULL, level); - dynType_logSetup(stdLog, NULL, level); - dynFunction_logSetup(stdLog, NULL, level); - dynInterface_logSetup(stdLog, NULL, level); } ~DynInterfaceTests() override { + celix_err_resetErrors(); } }; @@ -220,3 +343,45 @@ TEST_F(DynInterfaceTests, testInvalid) { testInvalid(); } + +TEST_F(DynInterfaceTests, testEmptyMethod) { + int status = 0; + dyn_interface_type *dynIntf = NULL; + FILE *desc = fopen("descriptors/example6.descriptor", "r"); + assert(desc != NULL); + status = dynInterface_parse(desc, &dynIntf); + ASSERT_EQ(0, status); + fclose(desc); + + const char *name = dynInterface_getName(dynIntf); + ASSERT_STREQ("calculator", name); + + checkInterfaceVersion(dynIntf,"1.0.0"); + + const struct methods_head* list = dynInterface_methods(dynIntf); + ASSERT_TRUE(list != NULL); + + int count = dynInterface_nrOfMethods(dynIntf); + ASSERT_EQ(0, count); + + dynInterface_destroy(dynIntf); +} + +TEST_F(DynInterfaceTests, testFindMethod) { + int status = 0; + dyn_interface_type *dynIntf = NULL; + FILE *desc = fopen("descriptors/example1.descriptor", "r"); + assert(desc != NULL); + status = dynInterface_parse(desc, &dynIntf); + ASSERT_EQ(0, status); + fclose(desc); + + const struct method_entry* mInfo = dynInterface_findMethod(dynIntf, "add(DD)D"); + ASSERT_TRUE(mInfo != NULL); + ASSERT_STREQ("add(DD)D", mInfo->id); + + mInfo = dynInterface_findMethod(dynIntf, "add(D)D"); + ASSERT_TRUE(mInfo == NULL); + + dynInterface_destroy(dynIntf); +} \ No newline at end of file diff --git a/libs/dfi/gtest/src/dyn_message_ei_tests.cc b/libs/dfi/gtest/src/dyn_message_ei_tests.cc new file mode 100644 index 000000000..2d09f6581 --- /dev/null +++ b/libs/dfi/gtest/src/dyn_message_ei_tests.cc @@ -0,0 +1,45 @@ +/* + * 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. + */ + +#include "dyn_message.h" +#include "celix_err.h" +#include "malloc_ei.h" + +#include + +class DynMessageErrorInjectionTestSuite : public ::testing::Test { +public: + DynMessageErrorInjectionTestSuite() = default; + ~DynMessageErrorInjectionTestSuite() override { + celix_ei_expect_calloc(nullptr, 0, nullptr); + } +}; + +TEST_F(DynMessageErrorInjectionTestSuite, ParseError) { + int status = 0; + dyn_message_type *dynMsg = NULL; + FILE *desc = fopen("descriptors/msg_example1.descriptor", "r"); + assert(desc != NULL); + //not enough memory for dyn_message_type + celix_ei_expect_calloc((void*) dynMessage_parse, 0, nullptr); + status = dynMessage_parse(desc, &dynMsg); + ASSERT_NE(0, status); + fclose(desc); + ASSERT_STREQ("Error allocating memory for dynamic message", celix_err_popLastError()); +} \ No newline at end of file diff --git a/libs/dfi/gtest/src/dyn_message_tests.cpp b/libs/dfi/gtest/src/dyn_message_tests.cpp index 4c1d242de..85e5578b1 100644 --- a/libs/dfi/gtest/src/dyn_message_tests.cpp +++ b/libs/dfi/gtest/src/dyn_message_tests.cpp @@ -34,35 +34,21 @@ extern "C" { #include "dyn_common.h" #include "dyn_message.h" - -#if NO_MEMSTREAM_AVAILABLE -#include "open_memstream.h" -#include "fmemopen.h" -#endif - -static void stdLog(void*, int level, const char *file, int line, const char *msg, ...) { - va_list ap; - const char *levels[5] = {"NIL", "ERROR", "WARNING", "INFO", "DEBUG"}; - fprintf(stderr, "%s: FILE:%s, LINE:%i, MSG:",levels[level], file, line); - va_start(ap, msg); - vfprintf(stderr, msg, ap); - fprintf(stderr, "\n"); - va_end(ap); -} +#include "celix_err.h" +#include "celix_version.h" static void checkMessageVersion(dyn_message_type* dynMsg, const char* v){ int status = 0; - char *version = NULL; - status = dynMessage_getVersionString(dynMsg, &version); - ASSERT_EQ(0, status); + const char* version = dynMessage_getVersionString(dynMsg); ASSERT_STREQ(v, version); - version_pt msgVersion = NULL, localMsgVersion = NULL; + const celix_version_t* msgVersion = nullptr; + celix_version_t* localMsgVersion = nullptr; int cmpVersion = -1; version_createVersionFromString(version,&localMsgVersion); - status = dynMessage_getVersion(dynMsg,&msgVersion); + msgVersion = dynMessage_getVersion(dynMsg); ASSERT_EQ(0, status); - version_compareTo(msgVersion,localMsgVersion,&cmpVersion); + cmpVersion = celix_version_compareTo(msgVersion, localMsgVersion); ASSERT_EQ(cmpVersion,0); version_destroy(localMsgVersion); @@ -78,26 +64,23 @@ static void msg_test1(void) { ASSERT_EQ(0, status); fclose(desc); - char *name = NULL; - status = dynMessage_getName(dynMsg, &name); + const char* name = dynMessage_getName(dynMsg); ASSERT_EQ(0, status); ASSERT_STREQ("poi", name); checkMessageVersion(dynMsg,"1.0.0"); - char *annVal = NULL; + const char* annVal = NULL; status = dynMessage_getAnnotationEntry(dynMsg, "classname", &annVal); ASSERT_EQ(0, status); ASSERT_STREQ("org.example.PointOfInterest", annVal); - char *nonExist = NULL; + const char* nonExist = NULL; status = dynMessage_getHeaderEntry(dynMsg, "nonExisting", &nonExist); ASSERT_TRUE(status != 0); ASSERT_TRUE(nonExist == NULL); - dyn_type *msgType = NULL; - status = dynMessage_getMessageType(dynMsg, &msgType); - ASSERT_EQ(0, status); + const dyn_type* msgType = dynMessage_getMessageType(dynMsg); ASSERT_TRUE(msgType != NULL); dynMessage_destroy(dynMsg); @@ -113,26 +96,23 @@ static void msg_test2(void) { ASSERT_EQ(0, status); fclose(desc); - char *name = NULL; - status = dynMessage_getName(dynMsg, &name); + const char* name = dynMessage_getName(dynMsg); ASSERT_EQ(0, status); ASSERT_STREQ("track", name); checkMessageVersion(dynMsg,"0.0.1"); - char *annVal = NULL; + const char* annVal = NULL; status = dynMessage_getAnnotationEntry(dynMsg, "classname", &annVal); ASSERT_EQ(0, status); ASSERT_STREQ("org.example.Track", annVal); - char *nonExist = NULL; + const char* nonExist = NULL; status = dynMessage_getHeaderEntry(dynMsg, "nonExisting", &nonExist); ASSERT_TRUE(status != 0); ASSERT_TRUE(nonExist == NULL); - dyn_type *msgType = NULL; - status = dynMessage_getMessageType(dynMsg, &msgType); - ASSERT_EQ(0, status); + const dyn_type* msgType = dynMessage_getMessageType(dynMsg); ASSERT_TRUE(msgType != NULL); dynMessage_destroy(dynMsg); @@ -147,26 +127,23 @@ static void msg_test3(void) { ASSERT_EQ(0, status); fclose(desc); - char *name = NULL; - status = dynMessage_getName(dynMsg, &name); + const char* name = dynMessage_getName(dynMsg); ASSERT_EQ(0, status); ASSERT_STREQ("logEntry", name); checkMessageVersion(dynMsg,"1.0.0"); - char *annVal = NULL; + const char* annVal = NULL; status = dynMessage_getAnnotationEntry(dynMsg, "classname", &annVal); ASSERT_EQ(0, status); ASSERT_STREQ("org.example.LogEntry", annVal); - char *nonExist = NULL; + const char* nonExist = NULL; status = dynMessage_getHeaderEntry(dynMsg, "nonExisting", &nonExist); ASSERT_TRUE(status != 0); ASSERT_TRUE(nonExist == NULL); - dyn_type *msgType = NULL; - status = dynMessage_getMessageType(dynMsg, &msgType); - ASSERT_EQ(0, status); + const dyn_type* msgType = dynMessage_getMessageType(dynMsg); ASSERT_TRUE(msgType != NULL); dynMessage_destroy(dynMsg); @@ -221,6 +198,23 @@ static void msg_invalid(void) { ASSERT_EQ(1, status); fclose(desc); + desc = fopen("descriptors/invalids/invalidMsgMissingName.descriptor", "r"); + assert(desc != NULL); + status = dynMessage_parse(desc, &dynMsg); + ASSERT_NE(0, status); + fclose(desc); + + desc = fopen("descriptors/invalids/invalidMsgType.descriptor", "r"); + assert(desc != NULL); + status = dynMessage_parse(desc, &dynMsg); + ASSERT_NE(0, status); + fclose(desc); + + desc = fopen("descriptors/invalids/invalidMsgMissingNewline.descriptor", "r"); + assert(desc != NULL); + status = dynMessage_parse(desc, &dynMsg); + ASSERT_NE(0, status); + fclose(desc); } } @@ -229,12 +223,9 @@ static void msg_invalid(void) { class DynMessageTests : public ::testing::Test { public: DynMessageTests() { - int level = 1; - dynCommon_logSetup(stdLog, NULL, level); - dynType_logSetup(stdLog, NULL, level); - dynMessage_logSetup(stdLog, NULL, level); } ~DynMessageTests() override { + celix_err_resetErrors(); } }; @@ -253,6 +244,7 @@ TEST_F(DynMessageTests, msg_test3) { TEST_F(DynMessageTests, msg_test4) { msg_test4(); + celix_err_printErrors(stderr, nullptr, nullptr); } TEST_F(DynMessageTests, msg_invalid) { diff --git a/libs/dfi/gtest/src/dyn_type_ei_tests.cc b/libs/dfi/gtest/src/dyn_type_ei_tests.cc new file mode 100644 index 000000000..cf5c80560 --- /dev/null +++ b/libs/dfi/gtest/src/dyn_type_ei_tests.cc @@ -0,0 +1,192 @@ +/* + * 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. + */ + +#include "dyn_type.h" +#include "celix_err.h" +#include "malloc_ei.h" +#include "stdio_ei.h" +#include "string_ei.h" + +#include +#include +#include + +class DynTypeErrorInjectionTestSuite : public ::testing::Test { +public: + DynTypeErrorInjectionTestSuite() { + } + + ~DynTypeErrorInjectionTestSuite() override { + celix_ei_expect_realloc(nullptr, 0, nullptr); + celix_ei_expect_strdup(nullptr, 0, nullptr); + celix_ei_expect_calloc(nullptr, 0, nullptr); + celix_ei_expect_fmemopen(nullptr, 0, nullptr); + celix_err_resetErrors(); + } + // delete other constructors and assign operators + DynTypeErrorInjectionTestSuite(DynTypeErrorInjectionTestSuite const&) = delete; + DynTypeErrorInjectionTestSuite(DynTypeErrorInjectionTestSuite&&) = delete; + DynTypeErrorInjectionTestSuite& operator=(DynTypeErrorInjectionTestSuite const&) = delete; + DynTypeErrorInjectionTestSuite& operator=(DynTypeErrorInjectionTestSuite&&) = delete; +}; + +TEST_F(DynTypeErrorInjectionTestSuite, ParseComplexTypeErrors) { + dyn_type *type = NULL; + const char* descriptor = "{D{DD b_1 b_2}I a b c}"; + + // fail to open memory as stream + celix_ei_expect_fmemopen((void*)dynType_parseWithStr, 0, nullptr); + int status = dynType_parseWithStr(descriptor, NULL, NULL, &type); + ASSERT_NE(0, status); + std::string msg = "Error creating mem stream for descriptor string. "; + msg += strerror(ENOMEM); + ASSERT_STREQ(msg.c_str(), celix_err_popLastError()); + + // fail to allocate dyn_type + celix_ei_expect_calloc((void*)dynType_parseWithStr, 2, nullptr); + status = dynType_parseWithStr(descriptor, NULL, NULL, &type); + ASSERT_NE(0, status); + ASSERT_STREQ("Error allocating memory for type", celix_err_popLastError()); + + // fail to duplicate type name + celix_ei_expect_strdup((void*)dynType_parseWithStr, 1, nullptr); + status = dynType_parseWithStr(descriptor, "hello", NULL, &type); + ASSERT_NE(0, status); + ASSERT_STREQ("Error strdup'ing name 'hello'", celix_err_popLastError()); + + // fail to allocate complex_type_entry + celix_ei_expect_calloc((void*)dynType_parseWithStr, 4, nullptr); + status = dynType_parseWithStr(descriptor, NULL, NULL, &type); + ASSERT_NE(0, status); + ASSERT_STREQ("Error allocating memory for complex_type_entry", celix_err_popLastError()); + + // fail to allocate ffi_type elements + celix_ei_expect_calloc((void*)dynType_parseWithStr, 4, nullptr, 4); + status = dynType_parseWithStr(descriptor, NULL, NULL, &type); + ASSERT_NE(0, status); + ASSERT_STREQ("Error allocating memory for ffi_type elements", celix_err_popLastError()); + + // fail to allocate complex types + celix_ei_expect_calloc((void*)dynType_parseWithStr, 4, nullptr, 5); + status = dynType_parseWithStr(descriptor, NULL, NULL, &type); + ASSERT_NE(0, status); + ASSERT_STREQ("Error allocating memory for complex types", celix_err_popLastError()); +} + +TEST_F(DynTypeErrorInjectionTestSuite, ParseNestedTypeErrors) { + dyn_type *type = NULL; + const char* descriptor = "Tnode={Lnode;Lnode; left right};{Lnode; head}"; + int status = 0; + + // fail to allocate type_entry + celix_ei_expect_calloc((void*)dynType_parseWithStr, 4, nullptr); + status = dynType_parseWithStr(descriptor, NULL, NULL, &type); + ASSERT_NE(0, status); + ASSERT_STREQ("Error allocating entry", celix_err_popLastError()); +} + +TEST_F(DynTypeErrorInjectionTestSuite, ParseEnumTypeErrors) { + dyn_type *type = NULL; + int rc = 0; + // fail to allocate meta_entry + celix_ei_expect_calloc((void*)dynType_parseWithStr, 4, nullptr, 1); + rc = dynType_parseWithStr("#v1=0;#v2=1;E", NULL, NULL, &type); + ASSERT_NE(0, rc); + celix_err_printErrors(stderr, nullptr, nullptr); +} + +TEST_F(DynTypeErrorInjectionTestSuite, AllocateErrors) { + celix_autoptr(dyn_type) type = NULL; + int rc = 0; + rc = dynType_parseWithStr("#v1=0;#v2=1;E", NULL, NULL, &type); + ASSERT_EQ(0, rc); + celix_ei_expect_calloc((void*)dynType_alloc, 0, nullptr); + void* buf = nullptr; + rc = dynType_alloc(type, &buf); + ASSERT_NE(0, rc); + ASSERT_STREQ("Error allocating memory for type 'E'", celix_err_popLastError()); +} + +TEST_F(DynTypeErrorInjectionTestSuite, SequenceAllocateError) { + struct double_sequence { + uint32_t cap; + uint32_t len; + double* buf; + }; + + dyn_type *type = NULL; + int rc = 0; + rc = dynType_parseWithStr("[D", NULL, NULL, &type); + ASSERT_EQ(0, rc); + + struct double_sequence *seq = NULL; + rc = dynType_alloc(type, (void **)&seq); + ASSERT_EQ(0, rc); + ASSERT_TRUE(seq != NULL); + celix_ei_expect_calloc((void*)dynType_sequence_alloc, 0, nullptr); + rc = dynType_sequence_alloc(type, seq, 1); + ASSERT_NE(0, rc); + ASSERT_STREQ("Error allocating memory for seq buf", celix_err_popLastError()); + + dynType_free(type, seq); + dynType_destroy(type); +} + +TEST_F(DynTypeErrorInjectionTestSuite, SequenceReserveError) { + struct double_sequence { + uint32_t cap; + uint32_t len; + double* buf; + }; + + dyn_type *type = NULL; + int rc = 0; + rc = dynType_parseWithStr("[D", NULL, NULL, &type); + ASSERT_EQ(0, rc); + + struct double_sequence *seq = NULL; + rc = dynType_alloc(type, (void **)&seq); + ASSERT_EQ(0, rc); + ASSERT_TRUE(seq != NULL); + celix_ei_expect_realloc((void*)dynType_sequence_reserve, 0, nullptr); + rc = dynType_sequence_reserve(type, seq, 1); + ASSERT_NE(0, rc); + ASSERT_STREQ("Error allocating memory for seq buf", celix_err_popLastError()); + + dynType_free(type, seq); + dynType_destroy(type); +} + +TEST_F(DynTypeErrorInjectionTestSuite, TextAllocateError) { + dyn_type *type = NULL; + int rc = 0; + rc = dynType_parseWithStr("t", NULL, NULL, &type); + ASSERT_EQ(0, rc); + ASSERT_EQ(DYN_TYPE_TEXT, dynType_type(type)); + + char** val = nullptr; + rc = dynType_alloc(type, (void**)&val); + ASSERT_EQ(0, rc); + celix_ei_expect_strdup((void*)dynType_text_allocAndInit, 0, nullptr); + rc = dynType_text_allocAndInit(type, val, "test"); + ASSERT_NE(0, rc); + ASSERT_STREQ("Cannot allocate memory for string", celix_err_popLastError()); + dynType_free(type, val); + dynType_destroy(type); +} \ No newline at end of file diff --git a/libs/dfi/gtest/src/dyn_type_tests.cpp b/libs/dfi/gtest/src/dyn_type_tests.cpp index 9cb23876a..701dae86a 100644 --- a/libs/dfi/gtest/src/dyn_type_tests.cpp +++ b/libs/dfi/gtest/src/dyn_type_tests.cpp @@ -16,32 +16,22 @@ * specific language governing permissions and limitations * under the License. */ - #include "gtest/gtest.h" +#include "celix_stdio_cleanup.h" +#include "celix_stdlib_cleanup.h" extern "C" { - #include - #include "dyn_common.h" #include "dyn_type.h" #include "celix_err.h" - static void stdLog(void*, int level, const char *file, int line, const char *msg, ...) { - va_list ap; - const char *levels[5] = {"NIL", "ERROR", "WARNING", "INFO", "DEBUG"}; - fprintf(stderr, "%s: FILE:%s, LINE:%i, MSG:",levels[level], file, line); - va_start(ap, msg); - vfprintf(stderr, msg, ap); - fprintf(stderr, "\n"); - va_end(ap); - } - static void runTest(const char *descriptorStr, const char *exName) { dyn_type *type; type = NULL; //printf("\n-- example %s with descriptor string '%s' --\n", exName, descriptorStr); int status = dynType_parseWithStr(descriptorStr, exName, NULL, &type); ASSERT_EQ(0, status); + ASSERT_STREQ(exName, dynType_getName(type)); //MEM check, to try to ensure no mem leaks/corruptions occur. int i; @@ -70,9 +60,9 @@ extern "C" { class DynTypeTests : public ::testing::Test { public: DynTypeTests() { - dynType_logSetup(stdLog, NULL, 1); } ~DynTypeTests() override { + celix_err_resetErrors(); } }; @@ -94,11 +84,17 @@ class DynTypeTests : public ::testing::Test { #define EX15 "Tsample={jDD time val1 val2};Tresult={jDlsample; time result sample};Lresult;" #define EX16 "Tpoi={BDD id lat lon};Lpoi;" #define EX17 "{#v1=0;#v2=1;E#v1=9;#v2=10;E enum1 enum2}" +#define EX18 "Ttext=t;ltext;" +#define EX19 "Tsample={DD vala valb};Tref=lsample;;lref;" +#define EX20 "TINTEGER=I;Tsample={DlINTEGER; vala valb};Tref=lsample;;lref;" #define CREATE_EXAMPLES_TEST(DESC) \ TEST_F(DynTypeTests, ParseTestExample ## DESC) { \ runTest(DESC, #DESC); \ - } + } \ + TEST_F(DynTypeTests, ParseTestExampleNoName ## DESC) { \ + runTest(DESC, nullptr); \ + } CREATE_EXAMPLES_TEST(EX1) CREATE_EXAMPLES_TEST(EX2) @@ -117,6 +113,9 @@ CREATE_EXAMPLES_TEST(EX14) CREATE_EXAMPLES_TEST(EX15) CREATE_EXAMPLES_TEST(EX16) CREATE_EXAMPLES_TEST(EX17) +CREATE_EXAMPLES_TEST(EX18) +CREATE_EXAMPLES_TEST(EX19) +CREATE_EXAMPLES_TEST(EX20) TEST_F(DynTypeTests, ParseRandomGarbageTest) { /* @@ -171,13 +170,20 @@ TEST_F(DynTypeTests, AssignTest1) { int32_t val1 = 2; int32_t val2 = 4; int32_t val3 = 8; + EXPECT_EQ(0, dynType_complex_indexForName(type, "a")); dynType_complex_setValueAt(type, 0, &inst, &val1); ASSERT_EQ(2, inst.a); + + EXPECT_EQ(1, dynType_complex_indexForName(type, "b")); dynType_complex_setValueAt(type, 1, &inst, &val2); ASSERT_EQ(4, inst.b); + + EXPECT_EQ(2, dynType_complex_indexForName(type, "c")); dynType_complex_setValueAt(type, 2, &inst, &val3); ASSERT_EQ(8, inst.c); + EXPECT_EQ(-1, dynType_complex_indexForName(type, nullptr)); + EXPECT_EQ(-1, dynType_complex_indexForName(type, "none")); dynType_destroy(type); } @@ -201,10 +207,8 @@ TEST_F(DynTypeTests, AssignTest2) { dynType_complex_setValueAt(type, 0, &inst, &a); ASSERT_EQ(2, inst.a); - void *loc = NULL; - dyn_type *subType = NULL; - dynType_complex_valLocAt(type, 1, (void *)&inst, &loc); - dynType_complex_dynTypeAt(type, 1, &subType); + (void)dynType_complex_valLocAt(type, 1, (void *)&inst); + const dyn_type* subType = dynType_complex_dynTypeAt(type, 1); dynType_complex_setValueAt(subType, 0, &inst.b, &b_a); ASSERT_EQ(1.1, inst.b.a); @@ -237,6 +241,17 @@ TEST_F(DynTypeTests, MetaInfoTest) { ASSERT_EQ(0, rc); + auto entries = dynType_metaEntries(type); + struct meta_entry* entry = NULL; + size_t nbEntries = 0; + TAILQ_FOREACH(entry, entries, entries) { + nbEntries++; + ASSERT_STREQ("a", entry->name); + ASSERT_STREQ("t", entry->value); + } + ASSERT_EQ(1, nbEntries); + + const char *val = NULL; val = dynType_getMetaInfo(type, "a"); ASSERT_TRUE(val != NULL); @@ -263,7 +278,7 @@ TEST_F(DynTypeTests, SequenceWithPointerTest) { struct val val; double c; double d; - long e; + int64_t e; }; struct item_sequence { @@ -274,7 +289,7 @@ TEST_F(DynTypeTests, SequenceWithPointerTest) { dyn_type *type = NULL; int rc = 0; - rc = dynType_parseWithStr("Tval={DD a b};Titem={Jtlval;DDJ a text val c d e};**[Litem;", NULL, NULL, &type); + rc = dynType_parseWithStr("Tval={DD a b};Titem={Jtlval;DDJ a text val c d e};[Litem;", NULL, NULL, &type); ASSERT_EQ(0, rc); struct item_sequence *seq = NULL; @@ -282,26 +297,127 @@ TEST_F(DynTypeTests, SequenceWithPointerTest) { ASSERT_EQ(0, rc); ASSERT_TRUE(seq != NULL); + dynType_sequence_init(type, seq); + rc = dynType_sequence_alloc(type, seq, 1); + ASSERT_EQ(0, rc); + struct item **loc = NULL; + rc = dynType_sequence_increaseLengthAndReturnLastLoc(type, seq, (void **)&loc); + ASSERT_EQ(0, rc); + ASSERT_EQ(loc, &seq->buf[0]); + ASSERT_EQ(sizeof(struct item), dynType_size(dynType_typedPointer_getTypedType(dynType_sequence_itemType(type)))); + rc = dynType_alloc(dynType_typedPointer_getTypedType(dynType_sequence_itemType(type)), (void**)loc); + ASSERT_EQ(0, rc); + ASSERT_EQ(seq->buf[0], *loc); dynType_free(type, seq); - /* + rc = dynType_sequence_alloc(type, nullptr, 1); + ASSERT_NE(0, rc); + ASSERT_STREQ("Error null sequence", celix_err_popLastError()); + dynType_destroy(type); +} - struct item_sequence *items = (struct item_sequence *) calloc(1,sizeof(struct item_sequence)); - items->buf = (struct item **) calloc(2, sizeof(struct item *)); - items->cap = 2; - items->len = 2; - items->buf[0] = (struct item *)calloc(1, sizeof(struct item)); - items->buf[0]->text = strdup("boe"); - items->buf[1] = (struct item *)calloc(1, sizeof(struct item)); - items->buf[1]->text = strdup("boe2"); - dynType_free(type, items); - */ +TEST_F(DynTypeTests, SequenceReserve) { + + struct val { + double a; + double b; + }; + + struct item { + int64_t a; + const char *text; + struct val val; + double c; + double d; + int64_t e; + }; + + struct item_sequence { + uint32_t cap; + uint32_t len; + struct item **buf; + }; + + dyn_type *type = NULL; + int rc = 0; + rc = dynType_parseWithStr("Tval={DD a b};Titem={Jtlval;DDJ a text val c d e};[Litem;", NULL, NULL, &type); + ASSERT_EQ(0, rc); + + struct item_sequence *seq = NULL; + + // sequence buffer should be zeroed + rc = dynType_alloc(type, (void **)&seq); + ASSERT_EQ(0, rc); + ASSERT_TRUE(seq != NULL); + rc = dynType_sequence_reserve(type, seq, 2); + ASSERT_EQ(0, rc); + ASSERT_EQ(nullptr, seq->buf[0]); + ASSERT_EQ(nullptr, seq->buf[1]); + ASSERT_EQ(2, seq->cap); + ASSERT_EQ(0, seq->len); + + // no need to expand capacity + rc = dynType_sequence_reserve(type, seq, 1); + ASSERT_EQ(0, rc); + dynType_free(type, seq); + + // try to reverse for null seq + rc = dynType_sequence_reserve(type, nullptr, 2); + ASSERT_NE(0, rc); + ASSERT_STREQ("Error null sequence", celix_err_popLastError()); dynType_destroy(type); } +TEST_F(DynTypeTests, FillSequenceTest) { + struct double_sequence { + uint32_t cap; + uint32_t len; + double* buf; + }; + + dyn_type *type = NULL; + int rc = 0; + rc = dynType_parseWithStr("[D", NULL, NULL, &type); + ASSERT_EQ(0, rc); + + struct double_sequence *seq = NULL; + rc = dynType_alloc(type, (void **)&seq); + ASSERT_EQ(0, rc); + ASSERT_TRUE(seq != NULL); + + rc = dynType_sequence_alloc(type, seq, 1); + ASSERT_EQ(0, rc); + + void* loc; + rc = dynType_sequence_increaseLengthAndReturnLastLoc(type, seq, (void **)&loc); + ASSERT_EQ(0, rc); + + double val = 2.0; + dynType_simple_setValue(dynType_sequence_itemType(type), loc, &val); + ASSERT_EQ(val, seq->buf[0]); + + rc = dynType_sequence_increaseLengthAndReturnLastLoc(type, seq, (void **)&loc); + ASSERT_NE(0, rc); + ASSERT_STREQ("Cannot increase sequence length beyond capacity (1)", celix_err_popLastError()); + + rc = dynType_sequence_reserve(type, seq, 2); + ASSERT_EQ(0, rc); + rc = dynType_sequence_locForIndex(type, seq, 1, (void **)&loc); + ASSERT_NE(0, rc); + ASSERT_STREQ("Requesting index (1) outsize defined length (1) but within capacity", celix_err_popLastError()); + + + rc = dynType_sequence_locForIndex(type, seq, 2, (void **)&loc); + ASSERT_NE(0, rc); + ASSERT_STREQ("Requested index (2) is greater than capacity (2) of sequence", celix_err_popLastError()); + + dynType_free(type, seq); + dynType_destroy(type); +} + TEST_F(DynTypeTests, EnumTest) { dyn_type *type = NULL; int rc = 0; @@ -312,6 +428,33 @@ TEST_F(DynTypeTests, EnumTest) { dynType_destroy(type); } + +TEST_F(DynTypeTests, PrintNullTypeTest) { + celix_autofree char* buf = nullptr; + size_t bufSize = 0; + celix_autoptr(FILE) result = open_memstream(&buf, &bufSize); + dynType_print(nullptr, result); + fflush(result); + ASSERT_STREQ("invalid type\n", buf); +} + +TEST_F(DynTypeTests, TextTest) { + dyn_type *type = NULL; + int rc = 0; + rc = dynType_parseWithStr("t", NULL, NULL, &type); + ASSERT_EQ(0, rc); + ASSERT_EQ(DYN_TYPE_TEXT, dynType_type(type)); + + char** val = nullptr; + rc = dynType_alloc(type, (void**)&val); + ASSERT_EQ(0, rc); + rc = dynType_text_allocAndInit(type, val, "test"); + ASSERT_EQ(0, rc); + ASSERT_STREQ("test", *val); + dynType_free(type, val); + dynType_destroy(type); +} + TEST_F(DynTypeTests, NrOfEntriesTest) { dyn_type *type = NULL; int rc = dynType_parseWithStr("{DD}", NULL, NULL, &type); @@ -333,6 +476,20 @@ TEST_F(DynTypeTests, ComplexHasEmptyName) { celix_err_printErrors(stderr, nullptr, nullptr); } +TEST_F(DynTypeTests, ComplexTypeMissingClosingBrace) { + dyn_type *type = NULL; + auto rc = dynType_parseWithStr(R"({II a b)", nullptr, nullptr, &type); + ASSERT_NE(0, rc); + ASSERT_STREQ("Error parsing complex type, expected '}'", celix_err_popLastError()); +} + +TEST_F(DynTypeTests, ComplexTypeWithMoreNamesThanFields) { + dyn_type *type = NULL; + auto rc = dynType_parseWithStr(R"({II a b c})", nullptr, nullptr, &type); + ASSERT_NE(0, rc); + ASSERT_STREQ("Error parsing complex type, expected '}'", celix_err_popLastError()); +} + TEST_F(DynTypeTests, SchemaEndsWithoutNullTerminator) { dyn_type *type = NULL; //ends with '-' @@ -341,17 +498,45 @@ TEST_F(DynTypeTests, SchemaEndsWithoutNullTerminator) { celix_err_printErrors(stderr, nullptr, nullptr); } +TEST_F(DynTypeTests, MetaInfoMissingName) { + dyn_type *type = NULL; + auto rc = dynType_parseWithStr(R"(#=1;)", nullptr, nullptr, &type); + ASSERT_NE(0, rc); + ASSERT_STREQ("Failed to parse meta properties", celix_err_popLastError()); + ASSERT_STREQ("Parsed empty name", celix_err_popLastError()); +} + +TEST_F(DynTypeTests, MetaInfoMissingEquality) { + dyn_type *type = NULL; + auto rc = dynType_parseWithStr(R"(#testMetaInfo 1;)", nullptr, nullptr, &type); + ASSERT_NE(0, rc); + celix_err_printErrors(stderr, nullptr, nullptr); +} + +TEST_F(DynTypeTests, MetaInfoMissingSemicolon) { + dyn_type *type = NULL; + auto rc = dynType_parseWithStr(R"(#testMetaInfo=1 )", nullptr, nullptr, &type); + ASSERT_NE(0, rc); + celix_err_printErrors(stderr, nullptr, nullptr); +} + TEST_F(DynTypeTests, MetaInfoMissingValue) { dyn_type *type = NULL; - auto rc = dynType_parseWithStr(R"(#testMetaInfo=)", nullptr, nullptr, &type); - ASSERT_EQ(1, rc); + auto rc = dynType_parseWithStr(R"(#testMetaInfo=;)", nullptr, nullptr, &type); + ASSERT_NE(0, rc); celix_err_printErrors(stderr, nullptr, nullptr); } TEST_F(DynTypeTests, ParseNestedTypeFailed) { dyn_type *type = NULL; + int rc; + //missing name + rc = dynType_parseWithStr(R"(T={DD a b};)", nullptr, nullptr, &type); + ASSERT_NE(0, rc); + ASSERT_STREQ("Parsed empty name", celix_err_popLastError()); + //missing '=' - auto rc = dynType_parseWithStr(R"(Ttype)", nullptr, nullptr, &type); + rc = dynType_parseWithStr(R"(Ttype)", nullptr, nullptr, &type); ASSERT_EQ(3, rc); celix_err_printErrors(stderr, nullptr, nullptr); @@ -368,8 +553,89 @@ TEST_F(DynTypeTests, ParseNestedTypeFailed) { TEST_F(DynTypeTests, ParseReferenceFailed) { dyn_type *type = NULL; + int rc; + // missing name + rc = dynType_parseWithStr(R"(Ttype={DD a b};l;)", nullptr, nullptr, &type); + ASSERT_NE(0, rc); + ASSERT_STREQ("Parsed empty name", celix_err_popLastError()); //missing ';' - auto rc = dynType_parseWithStr(R"(Ttype={DD a b};ltype)", nullptr, nullptr, &type); + rc = dynType_parseWithStr(R"(Ttype={DD a b};ltype)", nullptr, nullptr, &type); ASSERT_EQ(3, rc); celix_err_printErrors(stderr, nullptr, nullptr); + //missing ';' + rc = dynType_parseWithStr(R"(Ttype={DD a b};Ltype)", nullptr, nullptr, &type); + ASSERT_EQ(3, rc); + celix_err_printErrors(stderr, nullptr, nullptr); +} + +TEST_F(DynTypeTests, ParseSequenceFailed) { + dyn_type *type = NULL; + int rc = 0; + rc = dynType_parseWithStr("Tval={DD a b};Titem={Jtlval;DDJ a text val c d e};**[Lite;", NULL, NULL, &type); + ASSERT_NE(0, rc); +} + +TEST_F(DynTypeTests, TrivialityTesT) { + dyn_type *type = NULL; + int rc = 0; + + // non pointer simple type is trivial + rc = dynType_parseWithStr("I", NULL, NULL, &type); + ASSERT_EQ(0, rc); + ASSERT_TRUE(dynType_isTrivial(type)); + dynType_destroy(type); + + // untyped pointer is non-trivial + rc = dynType_parseWithStr("P", NULL, NULL, &type); + ASSERT_EQ(0, rc); + ASSERT_FALSE(dynType_isTrivial(type)); + dynType_destroy(type); + + // reference to non-pointer simple type is trivial + rc = dynType_parseWithStr("Ttype=I;ltype;", NULL, NULL, &type); + ASSERT_EQ(0, rc); + ASSERT_TRUE(dynType_isTrivial(type)); + dynType_destroy(type); + + // typed pointer is non-trivial + rc = dynType_parseWithStr("*D", NULL, NULL, &type); + ASSERT_EQ(0, rc); + ASSERT_FALSE(dynType_isTrivial(type)); + dynType_destroy(type); + + // sequence type is non-trivial + rc = dynType_parseWithStr("[I", NULL, NULL, &type); + ASSERT_EQ(0, rc); + EXPECT_FALSE(dynType_isTrivial(type)); + dynType_destroy(type); + + // text is non-trivial + rc = dynType_parseWithStr("t", NULL, NULL, &type); + ASSERT_EQ(0, rc); + EXPECT_FALSE(dynType_isTrivial(type)); + dynType_destroy(type); + + // a complex consisting of non-pointer scalars is trivial + rc = dynType_parseWithStr("{II a b}", NULL, NULL, &type); + ASSERT_EQ(0, rc); + EXPECT_TRUE(dynType_isTrivial(type)); + dynType_destroy(type); + + // a complex having a pointer scalar is non-trivial + rc = dynType_parseWithStr("{IP a b}", NULL, NULL, &type); + ASSERT_EQ(0, rc); + EXPECT_FALSE(dynType_isTrivial(type)); + dynType_destroy(type); + + // a complex consisting of non-pointer scalar and trivial complex is trivial + rc = dynType_parseWithStr("{II{II b c}}", NULL, NULL, &type); + ASSERT_EQ(0, rc); + EXPECT_TRUE(dynType_isTrivial(type)); + dynType_destroy(type); + + // a complex consisting of non-pointer scalar and non-trivial complex is non-trivial + rc = dynType_parseWithStr("{II{IP b c}}", NULL, NULL, &type); + ASSERT_EQ(0, rc); + EXPECT_FALSE(dynType_isTrivial(type)); + dynType_destroy(type); } \ No newline at end of file diff --git a/libs/dfi/gtest/src/json_rpc_ei_tests.cc b/libs/dfi/gtest/src/json_rpc_ei_tests.cc new file mode 100644 index 000000000..dd0851fc4 --- /dev/null +++ b/libs/dfi/gtest/src/json_rpc_ei_tests.cc @@ -0,0 +1,189 @@ +/* + * 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. + */ + +#include "json_rpc.h" +#include "json_rpc_test.h" +#include "json_serializer.h" +#include "dyn_type.h" +#include "dyn_function.h" +#include "celix_err.h" +#include "jansson_ei.h" +#include "malloc_ei.h" + +#include + +class JsonRpcErrorInjectionTestSuite : public ::testing::Test { +public: + JsonRpcErrorInjectionTestSuite() { + + } + ~JsonRpcErrorInjectionTestSuite() override { + celix_ei_expect_json_dumps(nullptr, 0, nullptr); + celix_ei_expect_json_object(nullptr, 0, nullptr); + celix_ei_expect_json_real(nullptr, 0, nullptr); + celix_ei_expect_calloc(nullptr, 0, nullptr); + celix_ei_expect_json_dumps(nullptr, 0, nullptr); + celix_ei_expect_json_array_append_new(nullptr, 0, -1); + celix_ei_expect_json_string(nullptr, 0, nullptr); + celix_ei_expect_json_array(nullptr, 0, nullptr); + celix_err_resetErrors(); + } +}; + +TEST_F(JsonRpcErrorInjectionTestSuite, prepareErrorTest) { + dyn_function_type *dynFunc = nullptr; + int rc = dynFunction_parseWithStr("add(#am=handle;PDD#am=pre;*D)N", nullptr, &dynFunc); + ASSERT_EQ(0, rc); + + char *result = nullptr; + + void *handle = nullptr; + double arg1 = 1.0; + double arg2 = 2.0; + + void *args[4]; + args[0] = &handle; + args[1] = &arg1; + args[2] = &arg2; + + celix_ei_expect_json_array((void*)jsonRpc_prepareInvokeRequest, 0, nullptr); + rc = jsonRpc_prepareInvokeRequest(dynFunc, "add", args, &result); + ASSERT_NE(0, rc); + EXPECT_STREQ("Error adding arguments array for 'add'", celix_err_popLastError()); + + celix_ei_expect_json_string((void*)jsonRpc_prepareInvokeRequest, 0, nullptr); + rc = jsonRpc_prepareInvokeRequest(dynFunc, "add", args, &result); + ASSERT_NE(0, rc); + EXPECT_STREQ("Error setting method name 'add'", celix_err_popLastError()); + + celix_ei_expect_json_array_append_new((void*)jsonRpc_prepareInvokeRequest, 0, -1); + rc = jsonRpc_prepareInvokeRequest(dynFunc, "add", args, &result); + ASSERT_NE(0, rc); + EXPECT_STREQ("Error adding argument (1) for 'add'", celix_err_popLastError()); + + celix_ei_expect_json_dumps((void*)jsonRpc_prepareInvokeRequest, 0, nullptr); + rc = jsonRpc_prepareInvokeRequest(dynFunc, "add", args, &result); + ASSERT_NE(0, rc); + + dynFunction_destroy(dynFunc); +} + +struct tst_serv { + void *handle; + int (*add)(void *, double, double, double *); + int (*sub)(void *, double, double, double *); + int (*sqrt)(void *, double, double *); + int (*stats)(void *, struct tst_seq, struct tst_StatsResult **); +}; + +TEST_F(JsonRpcErrorInjectionTestSuite, preallocationFailureTest) { + dyn_interface_type *intf = nullptr; + FILE *desc = fopen("descriptors/example1.descriptor", "r"); + ASSERT_TRUE(desc != nullptr); + int rc = dynInterface_parse(desc, &intf); + ASSERT_EQ(0, rc); + fclose(desc); + + char *result = nullptr; + tst_serv serv {nullptr, add, nullptr, nullptr, nullptr}; + + celix_ei_expect_calloc((void*)dynType_alloc, 0, nullptr); + rc = jsonRpc_call(intf, &serv, R"({"m":"add(DD)D", "a": [1.0,2.0]})", &result); + ASSERT_NE(0, rc); + EXPECT_EQ(nullptr, result); + EXPECT_STREQ("Error allocating memory for pre-allocated output argument of add(DD)D", celix_err_popLastError()); + EXPECT_STREQ("Error allocating memory for type 'D'", celix_err_popLastError()); + dynInterface_destroy(intf); +} + +TEST_F(JsonRpcErrorInjectionTestSuite, preOutParamterSerializationFailureTest) { + dyn_interface_type *intf = nullptr; + FILE *desc = fopen("descriptors/example1.descriptor", "r"); + ASSERT_TRUE(desc != nullptr); + int rc = dynInterface_parse(desc, &intf); + ASSERT_EQ(0, rc); + fclose(desc); + + char *result = nullptr; + tst_serv serv {nullptr, add, nullptr, nullptr, nullptr}; + + celix_ei_expect_json_real((void*)jsonSerializer_serializeJson, 2, nullptr); + rc = jsonRpc_call(intf, &serv, R"({"m":"add(DD)D", "a": [1.0,2.0]})", &result); + ASSERT_NE(0, rc); + EXPECT_EQ(nullptr, result); + EXPECT_STREQ("Error serializing result for add(DD)D", celix_err_popLastError()); + dynInterface_destroy(intf); +} + +TEST_F(JsonRpcErrorInjectionTestSuite, outParamterSerializationFailureTest) { + dyn_interface_type *intf = nullptr; + FILE *desc = fopen("descriptors/example1.descriptor", "r"); + ASSERT_TRUE(desc != nullptr); + int rc = dynInterface_parse(desc, &intf); + ASSERT_EQ(0, rc); + fclose(desc); + + char *result = nullptr; + tst_serv serv {nullptr, nullptr, nullptr, nullptr, stats}; + + celix_ei_expect_json_object((void*)jsonSerializer_serializeJson, 3, nullptr); + rc = jsonRpc_call(intf, &serv, R"({"m":"stats([D)LStatsResult;", "a": [[1.0,2.0]]})", &result); + ASSERT_NE(0, rc); + EXPECT_STREQ("Error serializing result for stats([D)LStatsResult;", celix_err_popLastError()); + + free(result); + dynInterface_destroy(intf); +} + +TEST_F(JsonRpcErrorInjectionTestSuite, responsePayloadGenerationErrorTest) { + dyn_interface_type *intf = nullptr; + FILE *desc = fopen("descriptors/example1.descriptor", "r"); + ASSERT_TRUE(desc != nullptr); + int rc = dynInterface_parse(desc, &intf); + ASSERT_EQ(0, rc); + fclose(desc); + + char *result = nullptr; + tst_serv serv {nullptr, add, nullptr, nullptr, nullptr}; + + celix_ei_expect_json_object((void*)jsonRpc_call, 0, nullptr); + rc = jsonRpc_call(intf, &serv, R"({"m":"add(DD)D", "a": [1.0,2.0]})", &result); + ASSERT_NE(0, rc); + EXPECT_EQ(nullptr, result); + EXPECT_STREQ("Error generating response payload for add(DD)D", celix_err_popLastError()); + dynInterface_destroy(intf); +} + +TEST_F(JsonRpcErrorInjectionTestSuite, responseRenderingErrorTest) { + dyn_interface_type *intf = nullptr; + FILE *desc = fopen("descriptors/example1.descriptor", "r"); + ASSERT_TRUE(desc != nullptr); + int rc = dynInterface_parse(desc, &intf); + ASSERT_EQ(0, rc); + fclose(desc); + + char *result = nullptr; + tst_serv serv {nullptr, add, nullptr, nullptr, nullptr}; + + celix_ei_expect_json_dumps((void*)jsonRpc_call, 0, nullptr); + rc = jsonRpc_call(intf, &serv, R"({"m":"add(DD)D", "a": [1.0,2.0]})", &result); + ASSERT_NE(0, rc); + EXPECT_EQ(nullptr, result); + dynInterface_destroy(intf); +} \ No newline at end of file diff --git a/libs/dfi/gtest/src/json_rpc_test.c b/libs/dfi/gtest/src/json_rpc_test.c new file mode 100644 index 000000000..57227b8f1 --- /dev/null +++ b/libs/dfi/gtest/src/json_rpc_test.c @@ -0,0 +1,67 @@ +/* + * 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. + */ + +#include "json_rpc_test.h" +#include "celix_compiler.h" +#include +#include +#include +#include +#include + +int add(void* handle CELIX_UNUSED, double a, double b, double *result) { + *result = a + b; + return 0; +} + +int stats(void* handle CELIX_UNUSED, struct tst_seq input, struct tst_StatsResult **out) { + assert(out != NULL); + assert(*out == NULL); + double total = 0.0; + unsigned int count = 0; + double max = DBL_MIN; + double min = DBL_MAX; + + unsigned int i; + for (i = 0; i max) { + max = input.buf[i]; + } + if (input.buf[i] < min) { + min = input.buf[i]; + } + } + + struct tst_StatsResult* result = (struct tst_StatsResult *)(calloc(1, sizeof(struct tst_StatsResult))); + if(count>0) { + result->average = total / count; + } + result->min = min; + result->max = max; + double* buf = (double *)(calloc(input.len, sizeof(double))); + memcpy(buf, input.buf, input.len * sizeof(double)); + result->input.len = input.len; + result->input.cap = input.len; + result->input.buf = buf; + + *out = result; + return 0; +} diff --git a/libs/dfi/gtest/src/json_rpc_test.h b/libs/dfi/gtest/src/json_rpc_test.h new file mode 100644 index 000000000..2e8deef44 --- /dev/null +++ b/libs/dfi/gtest/src/json_rpc_test.h @@ -0,0 +1,48 @@ +/* + * 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. + */ +#ifndef CELIX_JSON_RPC_TEST_H +#define CELIX_JSON_RPC_TEST_H +#ifdef __cplusplus +extern "C" { +#endif + +#include + +struct tst_seq { + uint32_t cap; + uint32_t len; + double *buf; +}; + + +//StatsResult={DDD[D average min max input} +struct tst_StatsResult { + double average; + double min; + double max; + struct tst_seq input; +}; + +int add(void*, double a, double b, double *result); +int stats(void*, struct tst_seq input, struct tst_StatsResult **out); + +#ifdef __cplusplus +} +#endif +#endif //CELIX_JSON_RPC_TEST_H diff --git a/libs/dfi/gtest/src/json_rpc_tests.cpp b/libs/dfi/gtest/src/json_rpc_tests.cpp index 32e45663f..260695a91 100644 --- a/libs/dfi/gtest/src/json_rpc_tests.cpp +++ b/libs/dfi/gtest/src/json_rpc_tests.cpp @@ -21,9 +21,9 @@ #include #include -#include "celix_err.h" extern "C" { +#include #include #include #include @@ -34,9 +34,15 @@ extern "C" { #include "dyn_common.h" #include "dyn_type.h" +#include "dyn_function.h" #include "json_serializer.h" #include "json_rpc.h" +#include "json_rpc_test.h" +#include "celix_compiler.h" #include "celix_errno.h" +#include "celix_err.h" + +#include void prepareTest(void) { @@ -149,7 +155,7 @@ extern "C" { rc = jsonRpc_handleReply(dynFunc, reply, args, &rsErrno); ASSERT_EQ(0, rc); ASSERT_EQ(0, rsErrno); - //ASSERT_EQ(2.2, result); + EXPECT_DOUBLE_EQ(2.2, result); dynFunction_destroy(dynFunc); } @@ -172,13 +178,6 @@ extern "C" { dynFunction_destroy(dynFunc); } - - - int add(void*, double a, double b, double *result) { - *result = a + b; - return 0; - } - int addFailed(void*, double , double , double *) { return CELIX_CUSTOMER_ERROR_MAKE(0,1);// return customer error } @@ -188,58 +187,6 @@ extern "C" { return 0; } - struct tst_seq { - uint32_t cap; - uint32_t len; - double *buf; - }; - - - //StatsResult={DDD[D average min max input} - struct tst_StatsResult { - double average; - double min; - double max; - struct tst_seq input; - }; - - - int stats(void*, struct tst_seq input, struct tst_StatsResult **out) { - assert(out != nullptr); - assert(*out == nullptr); - double total = 0.0; - unsigned int count = 0; - auto max = DBL_MIN; - auto min = DBL_MAX; - - unsigned int i; - for (i = 0; i max) { - max = input.buf[i]; - } - if (input.buf[i] < min) { - min = input.buf[i]; - } - } - - auto result = static_cast(calloc(1, sizeof(tst_StatsResult))); - if(count>0) { - result->average = total / count; - } - result->min = min; - result->max = max; - auto buf = static_cast(calloc(input.len, sizeof(double))); - memcpy(buf, input.buf, input.len * sizeof(double)); - result->input.len = input.len; - result->input.cap = input.len; - result->input.buf = buf; - - *out = result; - return 0; - } - struct item { double a; double b; @@ -335,14 +282,39 @@ extern "C" { tst_serv serv {nullptr, addFailed, nullptr, nullptr, nullptr}; rc = jsonRpc_call(intf, &serv, R"({)", &result); + EXPECT_STREQ("Got json error: string or '}' expected near end of file", celix_err_popLastError()); ASSERT_EQ(1, rc); + celix_err_resetErrors(); rc = jsonRpc_call(intf, &serv, R"({"a": [1.0,2.0]})", &result); + EXPECT_STREQ("Error getting method signature", celix_err_popLastError()); + ASSERT_EQ(1, rc); + celix_err_resetErrors(); + + //request missing argument + rc = jsonRpc_call(intf, &serv, R"({"m":"stats([D)LStatsResult;"})", &result); + EXPECT_STREQ("Error getting arguments array for stats([D)LStatsResult;", celix_err_popLastError()); + ASSERT_EQ(1, rc); + celix_err_resetErrors(); + + //request non-array argument + rc = jsonRpc_call(intf, &serv, R"({"m":"stats([D)LStatsResult;", "a": "hello"})", &result); + EXPECT_STREQ("Error getting arguments array for stats([D)LStatsResult;", celix_err_popLastError()); + ASSERT_EQ(1, rc); + celix_err_resetErrors(); + + // argument number mismatch + rc = jsonRpc_call(intf, &serv, R"({"m":"stats([D)LStatsResult;", "a": []})", &result); ASSERT_EQ(1, rc); + EXPECT_STREQ("Wrong number of standard arguments for stats([D)LStatsResult;. Expected 1, got 0", celix_err_popLastError()); + celix_err_resetErrors(); //request argument type mismatch rc = jsonRpc_call(intf, &serv, R"({"m":"stats([D)LStatsResult;", "a": [1.0]})", &result); ASSERT_EQ(1, rc); + EXPECT_STREQ("Error deserializing argument 1 for stats([D)LStatsResult;", celix_err_popLastError()); + EXPECT_STREQ("Expected json array type got '4'", celix_err_popLastError()); + celix_err_resetErrors(); dynInterface_destroy(intf); } @@ -355,17 +327,8 @@ extern "C" { ASSERT_EQ(0, rc); fclose(desc); - struct methods_head *head; - dynInterface_methods(intf, &head); - dyn_function_type *func = nullptr; - struct method_entry *entry = nullptr; - TAILQ_FOREACH(entry, head, entries) { - if (strcmp(entry->name, "stats") == 0) { - func = entry->dynFunc; - break; - } - } - ASSERT_TRUE(func != nullptr); + dyn_function_type *func = dynInterface_findMethod(intf, "stats([D)LStatsResult;")->dynFunc; + assert(func != nullptr); const char *reply = R"({"r":{"input":[1.0,2.0],"max":2.0,"average":1.5,"min":1.0}})"; @@ -397,17 +360,8 @@ extern "C" { ASSERT_EQ(0, rc); fclose(desc); - struct methods_head *head; - dynInterface_methods(intf, &head); - dyn_function_type *func = nullptr; - struct method_entry *entry = nullptr; - TAILQ_FOREACH(entry, head, entries) { - if (strcmp(entry->name, "stats") == 0) { - func = entry->dynFunc; - break; - } - } - ASSERT_TRUE(func != nullptr); + dyn_function_type *func = dynInterface_findMethod(intf, "stats([D)LStatsResult;")->dynFunc; + assert(func != nullptr); const char *reply = R"({})"; @@ -436,17 +390,9 @@ extern "C" { ASSERT_EQ(0, rc); fclose(desc); - struct methods_head *head; - dynInterface_methods(intf, &head); - dyn_function_type *func = nullptr; - struct method_entry *entry = nullptr; - TAILQ_FOREACH(entry, head, entries) { - if (strcmp(entry->name, "add") == 0) { - func = entry->dynFunc; - break; - } - } - ASSERT_TRUE(func != nullptr); + + dyn_function_type *func = dynInterface_findMethod(intf, "add(DD)D")->dynFunc; + assert(func != nullptr); const char *reply = R"({"e":33554433})"; @@ -482,6 +428,7 @@ extern "C" { rc = jsonRpc_call(intf, &serv, R"({"m":"unknown", "a": [1.0,2.0]})", &result); ASSERT_EQ(1, rc); + EXPECT_STREQ("Cannot find method with sig 'unknown'", celix_err_popLastError()); dynInterface_destroy(intf); } @@ -494,20 +441,8 @@ extern "C" { ASSERT_EQ(0, rc); fclose(desc); - struct methods_head *head; - dynInterface_methods(intf, &head); - dyn_function_type *func = nullptr; - struct method_entry *entry = nullptr; - TAILQ_FOREACH(entry, head, entries) { - if (strcmp(entry->name, "example1") == 0) { - func = entry->dynFunc; - break; - } - } - ASSERT_TRUE(func != nullptr); - - //dyn_type *arg = dynFunction_argumentTypeForIndex(func, 1); - //dynType_print(arg, stdout); + dyn_function_type *func = dynInterface_findMethod(intf, "example1")->dynFunc; + assert(func != nullptr); const char *reply = R"({"r":[{"a":1.0,"b":1.5},{"a":2.0,"b":2.5}]})"; @@ -547,17 +482,8 @@ extern "C" { ASSERT_EQ(0, rc); fclose(desc); - struct methods_head *head; - dynInterface_methods(intf, &head); - dyn_function_type *func = nullptr; - struct method_entry *entry = nullptr; - TAILQ_FOREACH(entry, head, entries) { - if (strcmp(entry->name, "action") == 0) { - func = entry->dynFunc; - break; - } - } - ASSERT_TRUE(func != nullptr); + dyn_function_type *func = dynInterface_findMethod(intf, "action(V)")->dynFunc; + assert(func != nullptr); const char *reply = R"({})"; @@ -571,95 +497,6 @@ extern "C" { dynInterface_destroy(intf); } - //Current only support one out argument - void handleTestMultiPreOut(void) { - dyn_interface_type *intf = nullptr; - FILE *desc = fopen("descriptors/invalids/multiOutArgs.descriptor", "r"); - ASSERT_TRUE(desc != nullptr); - int rc = dynInterface_parse(desc, &intf); - ASSERT_EQ(0, rc); - fclose(desc); - - struct methods_head *head; - dynInterface_methods(intf, &head); - dyn_function_type *func = nullptr; - struct method_entry *entry = nullptr; - TAILQ_FOREACH(entry, head, entries) { - if (strcmp(entry->name, "multiPreOut") == 0) { - func = entry->dynFunc; - break; - } - } - ASSERT_TRUE(func != nullptr); - - const char *reply = R"({"r":2.0})"; - - void *args[3]; - args[0] = nullptr; - args[1] = nullptr; - args[2] = nullptr; - - double result1 = 0; - void *out = &result1; - args[1] = &out; - int result2 = 0; - void *out2 = &result2; - args[2] = &out2; - - int rsErrno = 0; - rc = jsonRpc_handleReply(func, reply, args, &rsErrno); - ASSERT_NE(0, rc); - celix_err_printErrors(stderr, nullptr, nullptr); - - dynInterface_destroy(intf); - } - - //Current only support one out argument - void handleTestMultiOut(void) { - dyn_interface_type *intf = nullptr; - FILE *desc = fopen("descriptors/invalids/multiOutArgs.descriptor", "r"); - ASSERT_TRUE(desc != nullptr); - int rc = dynInterface_parse(desc, &intf); - ASSERT_EQ(0, rc); - fclose(desc); - - struct methods_head *head; - dynInterface_methods(intf, &head); - dyn_function_type *func = nullptr; - struct method_entry *entry = nullptr; - TAILQ_FOREACH(entry, head, entries) { - if (strcmp(entry->name, "multiOut") == 0) { - func = entry->dynFunc; - break; - } - } - ASSERT_TRUE(func != nullptr); - - const char *reply = R"({"r":2.0})"; - - void *args[3]; - args[0] = nullptr; - args[1] = nullptr; - args[2] = nullptr; - - double *result1 = nullptr; - void *out = &result1; - args[1] = &out; - int *result2 = nullptr; - void *out2 = &result2; - args[2] = &out2; - - int rsErrno = 0; - rc = jsonRpc_handleReply(func, reply, args, &rsErrno); - ASSERT_NE(0, rc); - celix_err_printErrors(stderr, nullptr, nullptr); - - dynInterface_destroy(intf); - } - - - - void callTestOutChar(void) { dyn_interface_type *intf = nullptr; FILE *desc = fopen("descriptors/example4.descriptor", "r"); @@ -689,18 +526,8 @@ extern "C" { ASSERT_EQ(0, rc); fclose(desc); - struct methods_head *head; - dynInterface_methods(intf, &head); - dyn_function_type *func = nullptr; - struct method_entry *entry = nullptr; - TAILQ_FOREACH(entry, head, entries) { - if (strcmp(entry->name, "getName") == 0) { - func = entry->dynFunc; - break; - } - } - - ASSERT_TRUE(func != nullptr); + dyn_function_type *func = dynInterface_findMethod(intf, "getName(V)t")->dynFunc; + assert(func != nullptr); const char *reply = R"({"r": "this is a test string"})"; char *result = nullptr; @@ -710,11 +537,9 @@ extern "C" { args[0] = nullptr; args[1] = &out; - if (func != nullptr) { // Check needed just to satisfy Coverity - int rsErrno = 0; - jsonRpc_handleReply(func, reply, args, &rsErrno); - ASSERT_EQ(0, rsErrno); - } + int rsErrno = 0; + jsonRpc_handleReply(func, reply, args, &rsErrno); + ASSERT_EQ(0, rsErrno); ASSERT_STREQ("this is a test string", result); @@ -722,26 +547,6 @@ extern "C" { dynInterface_destroy(intf); } - void handleTestPreChar(void) { - dyn_function_type *dynFunc = nullptr; - int rc = dynFunction_parseWithStr("getName(#am=handle;P#am=pre;t)N", nullptr, &dynFunc); - ASSERT_EQ(0, rc); - - const char *reply = "{\"r\":\"this is a test pre string\"}"; - char result[32] = {0}; - void *out = result; - void *args[2]; - args[0] = nullptr; - args[1] = &out; - int rsErrno = 0; - rc = jsonRpc_handleReply(dynFunc, reply, args, &rsErrno); - ASSERT_EQ(0, rc); - ASSERT_EQ(0, rsErrno); - ASSERT_STREQ("this is a test pre string", result); - - dynFunction_destroy(dynFunc); - } - void callTestChar(void) { dyn_interface_type *intf = nullptr; FILE *desc = fopen("descriptors/example4.descriptor", "r"); @@ -784,159 +589,6 @@ extern "C" { dynInterface_destroy(intf); } - - enum example6_enum{ - v1 = 1, - v2 = 2, - }; - - struct tst_CptData { - double d; - char *t; - struct tst_seq s; - enum example6_enum e; - }; - - struct tst_serv_example6 { - void *handle; - int (*cpt)(void *, struct tst_CptData *input, struct tst_CptData *output); - }; - - void testRequestBackwardCompatibility(void) { - dyn_interface_type *intf = nullptr; - FILE *desc = fopen("descriptors/example6.descriptor", "r"); - ASSERT_TRUE(desc != nullptr); - int rc = dynInterface_parse(desc, &intf); - ASSERT_EQ(0, rc); - fclose(desc); - - - char *result = nullptr; - - tst_serv_example6 serv {nullptr, nullptr}; - - serv.cpt = [](void *, struct tst_CptData *input, struct tst_CptData *)->int { - EXPECT_EQ(input->d , 0.0); - EXPECT_EQ(input->t , nullptr); - EXPECT_EQ(input->s.len , 0); - EXPECT_EQ(input->s.cap , 0); - EXPECT_EQ(input->s.buf , nullptr); - EXPECT_EQ(input->e , 0); - return 0; - }; - rc = jsonRpc_call(intf, &serv, R"({"m": "compatibility", "a": [{}]})", &result); - ASSERT_EQ(0, rc); - free(result); - - serv.cpt = [](void *, struct tst_CptData *input, struct tst_CptData *)->int { - EXPECT_EQ(input->d , 1.0); - EXPECT_EQ(input->t , nullptr); - EXPECT_EQ(input->s.len , 0); - EXPECT_EQ(input->s.cap , 0); - EXPECT_EQ(input->s.buf , nullptr); - EXPECT_EQ(input->e , 0); - return 0; - }; - rc = jsonRpc_call(intf, &serv, R"({"m": "compatibility", "a": [{"d":1.0}]})", &result); - ASSERT_EQ(0, rc); - free(result); - - serv.cpt = [](void *, struct tst_CptData *input, struct tst_CptData *)->int { - EXPECT_EQ(input->d , 1.0); - EXPECT_STREQ(input->t , "hello compatibility"); - EXPECT_EQ(input->s.len , 0); - EXPECT_EQ(input->s.cap , 0); - EXPECT_EQ(input->s.buf , nullptr); - EXPECT_EQ(input->e , 0); - return 0; - }; - rc = jsonRpc_call(intf, &serv, R"({"m": "compatibility", "a": [{"d":1.0, "t":"hello compatibility"}]})", &result); - ASSERT_EQ(0, rc); - free(result); - - serv.cpt = [](void *, struct tst_CptData *input, struct tst_CptData *)->int { - EXPECT_EQ(input->d , 1.0); - EXPECT_STREQ(input->t , "hello compatibility"); - EXPECT_EQ(input->s.len , 3); - EXPECT_EQ(input->s.cap , 3); - EXPECT_EQ(input->s.buf[0] , 1.0); - EXPECT_EQ(input->s.buf[1] , 2.0); - EXPECT_EQ(input->s.buf[2] , 3.0); - EXPECT_EQ(input->e , 0); - return 0; - }; - rc = jsonRpc_call(intf, &serv, R"({"m": "compatibility", "a": [{"d":1.0, "t":"hello compatibility", "s":[1.0,2.0,3.0]}]})", &result); - ASSERT_EQ(0, rc); - free(result); - - - serv.cpt = [](void *, struct tst_CptData *input, struct tst_CptData *)->int { - EXPECT_EQ(input->d , 1.0); - EXPECT_STREQ(input->t , "hello compatibility"); - EXPECT_EQ(input->s.len , 3); - EXPECT_EQ(input->s.cap , 3); - EXPECT_EQ(input->s.buf[0] , 1.0); - EXPECT_EQ(input->s.buf[1] , 2.0); - EXPECT_EQ(input->s.buf[2] , 3.0); - EXPECT_EQ(input->e , v2); - return 0; - }; - rc = jsonRpc_call(intf, &serv, R"({"m": "compatibility", "a": [{"d":1.0, "t":"hello compatibility", "s":[1.0,2.0,3.0], "e":"v2"}]})", &result); - ASSERT_EQ(0, rc); - free(result); - - - dynInterface_destroy(intf); - } - - void testResponseForwardCompatibility(void) { - dyn_interface_type *intf = nullptr; - FILE *desc = fopen("descriptors/example6.descriptor", "r"); - ASSERT_TRUE(desc != nullptr); - int rc = dynInterface_parse(desc, &intf); - ASSERT_EQ(0, rc); - fclose(desc); - - struct methods_head *head; - dynInterface_methods(intf, &head); - dyn_function_type *func = nullptr; - struct method_entry *entry = nullptr; - TAILQ_FOREACH(entry, head, entries) { - if (strcmp(entry->name, "compatibility") == 0) { - func = entry->dynFunc; - break; - } - } - ASSERT_TRUE(func != nullptr); - - struct tst_CptData *cptData{nullptr}; - void *out = &cptData; - - void *args[3]; - args[0] = nullptr; - args[1] = nullptr; - args[2] = &out; - int rsErrno = 0; - - //provider has more reply - rc = jsonRpc_handleReply(func, R"({"r":{"d":1.0, "t":"hello compatibility", "s":[1.0,2.0,3.0], "e":"v1", "e2":"v2"}})", args, &rsErrno); - EXPECT_EQ(0, rc); - EXPECT_EQ(0, rsErrno); - EXPECT_NE(cptData , nullptr); - EXPECT_EQ(cptData->d , 1.0); - EXPECT_STREQ(cptData->t , "hello compatibility"); - EXPECT_EQ(cptData->s.len , 3); - EXPECT_EQ(cptData->s.cap , 3); - EXPECT_EQ(cptData->s.buf[0] , 1.0); - EXPECT_EQ(cptData->s.buf[1] , 2.0); - EXPECT_EQ(cptData->s.buf[2] , 3.0); - EXPECT_EQ(cptData->e , v1); - free(cptData->t); - free(cptData->s.buf); - free(cptData); - - dynInterface_destroy(intf); - } } class JsonRpcTests : public ::testing::Test { @@ -945,6 +597,7 @@ class JsonRpcTests : public ::testing::Test { } ~JsonRpcTests() override { + celix_err_resetErrors(); } }; @@ -969,6 +622,24 @@ TEST_F(JsonRpcTests, handleTestPre) { handleTestPre(); } +TEST_F(JsonRpcTests, handleTestNullPre) { + dyn_function_type *dynFunc = nullptr; + int rc = dynFunction_parseWithStr("add(#am=handle;PDD#am=pre;*D)N", nullptr, &dynFunc); + ASSERT_EQ(0, rc); + + const char *reply = "{\"r\":2.2}"; + double *out = NULL; + void *args[4]; + args[3] = &out; + int rsErrno = 0; + rc = jsonRpc_handleReply(dynFunc, reply, args, &rsErrno); + ASSERT_EQ(0, rc); + ASSERT_EQ(0, rsErrno); + //ASSERT_EQ(2.2, result); + + dynFunction_destroy(dynFunc); +} + TEST_F(JsonRpcTests, handleTestInvalidReply) { handleTestInvalidReply(); } @@ -977,10 +648,105 @@ TEST_F(JsonRpcTests, handleTestOut) { handleTestOut(); } +TEST_F(JsonRpcTests, handleTestNullOutResult) { + dyn_interface_type *intf = nullptr; + FILE *desc = fopen("descriptors/example1.descriptor", "r"); + ASSERT_TRUE(desc != nullptr); + int rc = dynInterface_parse(desc, &intf); + ASSERT_EQ(0, rc); + fclose(desc); + + dyn_function_type *func = dynInterface_findMethod(intf, "stats([D)LStatsResult;")->dynFunc; + ASSERT_TRUE(func != nullptr); + + const char *reply = R"({"r":null})"; + + void *args[3]; + args[0] = nullptr; + args[1] = nullptr; + args[2] = nullptr; + + struct tst_StatsResult *result = nullptr; + void *out = &result; + args[2] = &out; + + int rsErrno = 0; + rc = jsonRpc_handleReply(func, reply, args, &rsErrno); + ASSERT_EQ(0, rc); + ASSERT_EQ(0, rsErrno); + ASSERT_EQ(nullptr, result); + dynInterface_destroy(intf); +} + +TEST_F(JsonRpcTests, handleTestNullOut) { + dyn_interface_type *intf = nullptr; + FILE *desc = fopen("descriptors/example1.descriptor", "r"); + ASSERT_TRUE(desc != nullptr); + int rc = dynInterface_parse(desc, &intf); + ASSERT_EQ(0, rc); + fclose(desc); + + dyn_function_type *func = dynInterface_findMethod(intf, "stats([D)LStatsResult;")->dynFunc; + assert(func != nullptr); + + const char *reply = R"({"r":{"input":[1.0,2.0],"max":2.0,"average":1.5,"min":1.0}})"; + + void *args[3]; + args[0] = nullptr; + args[1] = nullptr; + args[2] = nullptr; + + void *out = nullptr; + args[2] = &out; + + int rsErrno = 0; + rc = jsonRpc_handleReply(func, reply, args, &rsErrno); + ASSERT_EQ(0, rc); + ASSERT_EQ(0, rsErrno); + dynInterface_destroy(intf); +} + +TEST_F(JsonRpcTests, callPreReference) { + dyn_interface_type *intf = nullptr; + FILE *desc = fopen("descriptors/example7.descriptor", "r"); + ASSERT_TRUE(desc != nullptr); + int rc = dynInterface_parse(desc, &intf); + ASSERT_EQ(0, rc); + fclose(desc); + + char *result = nullptr; + tst_serv serv {nullptr, add, nullptr, nullptr, nullptr}; + + rc = jsonRpc_call(intf, &serv, R"({"m":"add(DD)D", "a": [1.0,2.0]})", &result); + ASSERT_EQ(0, rc); + ASSERT_TRUE(strstr(result, "3.0") != nullptr); + + free(result); + dynInterface_destroy(intf); +} + TEST_F(JsonRpcTests, callPre) { callTestPreAllocated(); } +TEST_F(JsonRpcTests, callPreWithMismatchedArgumentNumber) { + dyn_interface_type *intf = nullptr; + FILE *desc = fopen("descriptors/example1.descriptor", "r"); + ASSERT_TRUE(desc != nullptr); + int rc = dynInterface_parse(desc, &intf); + ASSERT_EQ(0, rc); + fclose(desc); + + char *result = nullptr; + tst_serv serv {nullptr, add, nullptr, nullptr, nullptr}; + + rc = jsonRpc_call(intf, &serv, R"({"m":"add(DD)D", "a": [1.0,2.0,3.0]})", &result); + EXPECT_NE(0, rc); + EXPECT_EQ(nullptr, result); + EXPECT_STREQ("Wrong number of standard arguments for add(DD)D. Expected 2, got 3", celix_err_popLastError()); + dynInterface_destroy(intf); +} + TEST_F(JsonRpcTests, callFailedPre) { callFailedTestPreAllocated(); } @@ -989,6 +755,50 @@ TEST_F(JsonRpcTests, callOut) { callTestOutput(); } +TEST_F(JsonRpcTests, callOutNullResult) { + dyn_interface_type *intf = nullptr; + FILE *desc = fopen("descriptors/example1.descriptor", "r"); + ASSERT_TRUE(desc != nullptr); + int rc = dynInterface_parse(desc, &intf); + ASSERT_EQ(0, rc); + fclose(desc); + + char *result = nullptr; + tst_serv serv {nullptr, nullptr, nullptr, nullptr, [](void*, struct tst_seq, struct tst_StatsResult **out)->int { + assert(out != nullptr); + assert(*out == nullptr); + *out = nullptr; + return 0; + }}; + + rc = jsonRpc_call(intf, &serv, R"({"m":"stats([D)LStatsResult;", "a": [[1.0,2.0]]})", &result); + ASSERT_EQ(0, rc); + + json_auto_t* replyJson = json_loads(result, JSON_DECODE_ANY, nullptr); + EXPECT_TRUE(json_is_null(json_object_get(replyJson, "r"))); + free(result); + dynInterface_destroy(intf); +} + +TEST_F(JsonRpcTests, callOutReference) { + dyn_interface_type *intf = nullptr; + FILE *desc = fopen("descriptors/example7.descriptor", "r"); + ASSERT_TRUE(desc != nullptr); + int rc = dynInterface_parse(desc, &intf); + ASSERT_EQ(0, rc); + fclose(desc); + + char *result = nullptr; + tst_serv serv {nullptr, nullptr, nullptr, nullptr, stats}; + + rc = jsonRpc_call(intf, &serv, R"({"m":"stats([D)LStatsResult;", "a": [[1.0,2.0]]})", &result); + ASSERT_EQ(0, rc); + ASSERT_TRUE(strstr(result, "1.5") != nullptr); + + free(result); + dynInterface_destroy(intf); +} + TEST_F(JsonRpcTests, callTestInvalidRequest) { callTestInvalidRequest(); } @@ -1001,16 +811,89 @@ TEST_F(JsonRpcTests, handleOutSeq) { handleTestOutputSequence(); } -TEST_F(JsonRpcTests, callTestOutChar) { +TEST_F(JsonRpcTests, callTestOutText) { callTestOutChar(); } -TEST_F(JsonRpcTests, handleOutChar) { +TEST_F(JsonRpcTests, callTestOutNullTextResult) { + dyn_interface_type *intf = nullptr; + FILE *desc = fopen("descriptors/example4.descriptor", "r"); + ASSERT_TRUE(desc != nullptr); + int rc = dynInterface_parse(desc, &intf); + ASSERT_EQ(0, rc); + fclose(desc); + + char *result = nullptr; + tst_serv_example4 serv {nullptr, [](void *, char** result)->int { + *result = nullptr; + return 0; + }, nullptr, nullptr}; + + rc = jsonRpc_call(intf, &serv, R"({"m": "getName(V)t", "a": []})", &result); + ASSERT_EQ(0, rc); + + json_auto_t* replyJson = json_loads(result, JSON_DECODE_ANY, nullptr); + EXPECT_TRUE(json_is_null(json_object_get(replyJson, "r"))); + free(result); + dynInterface_destroy(intf); +} + +TEST_F(JsonRpcTests, handleOutText) { handleTestOutChar(); } -TEST_F(JsonRpcTests, handlePreChar) { - handleTestPreChar(); +TEST_F(JsonRpcTests, handleNullOutTextResult) { + dyn_interface_type *intf = nullptr; + FILE *desc = fopen("descriptors/example4.descriptor", "r"); + ASSERT_TRUE(desc != nullptr); + int rc = dynInterface_parse(desc, &intf); + ASSERT_EQ(0, rc); + fclose(desc); + + dyn_function_type *func = dynInterface_findMethod(intf, "getName(V)t")->dynFunc; + assert(func != nullptr); + + const char *reply = R"({"r":null})"; + char *result = nullptr; + void *out = &result; + + void *args[2]; + args[0] = nullptr; + args[1] = &out; + + int rsErrno = 0; + jsonRpc_handleReply(func, reply, args, &rsErrno); + ASSERT_EQ(0, rsErrno); + EXPECT_EQ(nullptr, result); + dynInterface_destroy(intf); +} + +TEST_F(JsonRpcTests, handleInvalidOutChar) { + + dyn_interface_type *intf = nullptr; + FILE *desc = fopen("descriptors/example4.descriptor", "r"); + ASSERT_TRUE(desc != nullptr); + int rc = dynInterface_parse(desc, &intf); + ASSERT_EQ(0, rc); + fclose(desc); + + dyn_function_type *func = dynInterface_findMethod(intf, "getName(V)t")->dynFunc; + assert(func != nullptr); + + const char *reply = R"({"r": 12345})"; + char *result = nullptr; + void *out = &result; + + void *args[2]; + args[0] = nullptr; + args[1] = &out; + + if (func != nullptr) { // Check needed just to satisfy Coverity + int rsErrno = 0; + int status = jsonRpc_handleReply(func, reply, args, &rsErrno); + EXPECT_NE(0, status); + } + dynInterface_destroy(intf); } TEST_F(JsonRpcTests, handleReplyError) { @@ -1025,14 +908,6 @@ TEST_F(JsonRpcTests, handleTestAction) { handleTestAction(); } -TEST_F(JsonRpcTests, handleTestMultiPreOut) { - handleTestMultiPreOut(); -} - -TEST_F(JsonRpcTests, handleTestMultiOut) { - handleTestMultiOut(); -} - TEST_F(JsonRpcTests, callTestChar) { callTestChar(); } @@ -1041,10 +916,20 @@ TEST_F(JsonRpcTests, callTestConstChar) { callTestConstChar(); } -TEST_F(JsonRpcTests, testRequestBackwardCompatibility) { - testRequestBackwardCompatibility(); -} - -TEST_F(JsonRpcTests, testResponseForwardCompatibility) { - testResponseForwardCompatibility(); -} +TEST_F(JsonRpcTests, callWithTooManyArguments) { + dyn_interface_type *intf = nullptr; + FILE *desc = fopen("descriptors/invalids/methodWithTooManyArgs.descriptor", "r"); + ASSERT_TRUE(desc != nullptr); + int rc = dynInterface_parse(desc, &intf); + ASSERT_EQ(0, rc); + fclose(desc); + + char *result = nullptr; + tst_serv serv {nullptr, add, nullptr, nullptr, nullptr}; + + rc = jsonRpc_call(intf, &serv, R"({"m":"add(DDDDDDDDDDDDDDD)D", "a": [1.0,2.0]})", &result); + ASSERT_NE(0, rc); + EXPECT_EQ(nullptr, result); + EXPECT_STREQ("Too many arguments for add(DDDDDDDDDDDDDDD)D: 17 > 16", celix_err_popLastError()); + dynInterface_destroy(intf); +} \ No newline at end of file diff --git a/libs/dfi/gtest/src/json_serializer_ei_tests.cc b/libs/dfi/gtest/src/json_serializer_ei_tests.cc new file mode 100644 index 000000000..4c7e9feba --- /dev/null +++ b/libs/dfi/gtest/src/json_serializer_ei_tests.cc @@ -0,0 +1,172 @@ +/* + * 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. + */ + +#include "json_serializer.h" +#include "dyn_type.h" +#include "celix_err.h" +#include "jansson_ei.h" +#include "malloc_ei.h" +#include "string_ei.h" + +#include +#include + +class JsonSerializerErrorInjectionTestSuite : public ::testing::Test { +public: + JsonSerializerErrorInjectionTestSuite() = default; + ~JsonSerializerErrorInjectionTestSuite() override { + celix_ei_expect_json_integer(nullptr, 0, nullptr); + celix_ei_expect_json_array_append_new(nullptr, 0, 0); + celix_ei_expect_json_array(nullptr, 0, nullptr); + celix_ei_expect_json_object_set_new(nullptr, 0, 0); + celix_ei_expect_json_object(nullptr, 0, nullptr); + celix_ei_expect_json_dumps(nullptr, 0, nullptr); + celix_ei_expect_json_array_size(nullptr, 0, 0); + celix_ei_expect_strdup(nullptr, 0, nullptr); + celix_ei_expect_calloc(nullptr, 0, nullptr); + celix_err_resetErrors(); + } +}; + +TEST_F(JsonSerializerErrorInjectionTestSuite, DeserilizationError) { + int rc; + dyn_type *type; + void *inst; + + type = nullptr; + inst = nullptr; + rc = dynType_parseWithStr("t", nullptr, nullptr, &type); + ASSERT_EQ(0, rc); + auto inputStr = R"("hello")"; + celix_ei_expect_calloc((void*) dynType_alloc, 0, nullptr); + rc = jsonSerializer_deserialize(type, inputStr, strlen(inputStr), &inst); + ASSERT_NE(0, rc); + EXPECT_STREQ("Error cannot deserialize json. Input is '\"hello\"'", celix_err_popLastError()); + EXPECT_STREQ("Error allocating memory for type 't'", celix_err_popLastError()); + dynType_destroy(type); + + type = nullptr; + inst = nullptr; + rc = dynType_parseWithStr("t", nullptr, nullptr, &type); + ASSERT_EQ(0, rc); + inputStr = R"("hello")"; + celix_ei_expect_strdup((void*) dynType_text_allocAndInit, 0, nullptr); + rc = jsonSerializer_deserialize(type, inputStr, strlen(inputStr), &inst); + ASSERT_NE(0, rc); + EXPECT_STREQ("Error cannot deserialize json. Input is '\"hello\"'", celix_err_popLastError()); + EXPECT_STREQ("Cannot allocate memory for string", celix_err_popLastError()); + dynType_destroy(type); + + + type = nullptr; + inst = nullptr; + rc = dynType_parseWithStr("[t", nullptr, nullptr, &type); + ASSERT_EQ(0, rc); + inputStr = R"(["hello", "world"])"; + celix_ei_expect_json_array_size((void*)jsonSerializer_deserializeJson, 3, (size_t)UINT32_MAX+1); + rc = jsonSerializer_deserialize(type, inputStr, strlen(inputStr), &inst); + ASSERT_NE(0, rc); + EXPECT_STREQ("Error cannot deserialize json. Input is '[\"hello\", \"world\"]'", celix_err_popLastError()); + EXPECT_STREQ("Error array size(4294967296) too large", celix_err_popLastError()); + dynType_destroy(type); + + + type = nullptr; + inst = nullptr; + rc = dynType_parseWithStr("[t", nullptr, nullptr, &type); + ASSERT_EQ(0, rc); + inputStr = R"(["hello", "world"])"; + celix_ei_expect_calloc((void*) dynType_sequence_alloc, 0, nullptr); + rc = jsonSerializer_deserialize(type, inputStr, strlen(inputStr), &inst); + ASSERT_NE(0, rc); + EXPECT_STREQ("Error cannot deserialize json. Input is '[\"hello\", \"world\"]'", celix_err_popLastError()); + EXPECT_STREQ("Error allocating memory for seq buf", celix_err_popLastError()); + dynType_destroy(type); +} + +struct test_struct { + double a; + double b; +}; + +struct test_seq { + uint32_t cap; + uint32_t len; + double* buf; +}; + +TEST_F(JsonSerializerErrorInjectionTestSuite, SerilizationError) { + dyn_type *type = nullptr; + char *result = nullptr; + int rc = dynType_parseWithStr(R"(#v1=1;#v2=2;E)", nullptr, nullptr, &type); + ASSERT_EQ(0, rc); + enum { + v1 = 1, + v2 = 2 + }enumVal = v2; + celix_ei_expect_json_dumps((void*)jsonSerializer_serialize, 0, nullptr); + rc = jsonSerializer_serialize(type, &enumVal, &result); + EXPECT_NE(0, rc); + EXPECT_EQ(nullptr, result); + dynType_destroy(type); + + test_struct ex {1.0, 2.0}; + rc = dynType_parseWithStr(R"({DD a b})", nullptr, nullptr, &type); + ASSERT_EQ(0, rc); + celix_ei_expect_json_object((void*)jsonSerializer_serialize, 3, nullptr); + rc = jsonSerializer_serialize(type, &ex, &result); + EXPECT_NE(0, rc); + EXPECT_EQ(nullptr, result); + dynType_destroy(type); + + rc = dynType_parseWithStr(R"({DD a b})", nullptr, nullptr, &type); + ASSERT_EQ(0, rc); + celix_ei_expect_json_object_set_new((void*)jsonSerializer_serialize, 3, -1); + rc = jsonSerializer_serialize(type, &ex, &result); + EXPECT_NE(0, rc); + EXPECT_EQ(nullptr, result); + dynType_destroy(type); + + double arr[1] = {1.0}; + test_seq seq {1, 1, arr}; + rc = dynType_parseWithStr(R"([D)", nullptr, nullptr, &type); + ASSERT_EQ(0, rc); + celix_ei_expect_json_array((void*)jsonSerializer_serialize, 3, nullptr); + rc = jsonSerializer_serialize(type, &seq, &result); + EXPECT_NE(0, rc); + EXPECT_EQ(nullptr, result); + dynType_destroy(type); + + rc = dynType_parseWithStr(R"([D)", nullptr, nullptr, &type); + ASSERT_EQ(0, rc); + celix_ei_expect_json_array_append_new((void*)jsonSerializer_serialize, 3, -1); + rc = jsonSerializer_serialize(type, &seq, &result); + EXPECT_NE(0, rc); + EXPECT_EQ(nullptr, result); + dynType_destroy(type); + + int32_t intVal = 12345; + rc = dynType_parseWithStr(R"(I)", nullptr, nullptr, &type); + ASSERT_EQ(0, rc); + celix_ei_expect_json_integer((void*)jsonSerializer_serialize, 2, nullptr); + rc = jsonSerializer_serialize(type, &intVal, &result); + EXPECT_NE(0, rc); + EXPECT_EQ(nullptr, result); + dynType_destroy(type); +} diff --git a/libs/dfi/gtest/src/json_serializer_tests.cpp b/libs/dfi/gtest/src/json_serializer_tests.cpp index 6aeb5ec35..20bdf7253 100644 --- a/libs/dfi/gtest/src/json_serializer_tests.cpp +++ b/libs/dfi/gtest/src/json_serializer_tests.cpp @@ -34,15 +34,7 @@ extern "C" { #include "json_serializer.h" #include "celix_err.h" -static void stdLog(void*, int level, const char *file, int line, const char *msg, ...) { - va_list ap; - const char *levels[5] = {"NIL", "ERROR", "WARNING", "INFO", "DEBUG"}; - fprintf(stderr, "%s: FILE:%s, LINE:%i, MSG:",levels[level], file, line); - va_start(ap, msg); - vfprintf(stderr, msg, ap); - fprintf(stderr, "\n"); - va_end(ap); -} +#include /*********** example 1 ************************/ /** struct type ******************************/ @@ -172,18 +164,24 @@ static const char *example5_input = "{ \ \"age\" : 44 \ },\ \"left\" : {\ + \"left\" : null,\ + \"right\" : null,\ \"value\" : {\ \"name\" : \"Victor\",\ \"age\" : 400 \ }\ - }\ + },\ + \"right\" : null\ },\ \"right\" : {\ + \"left\" : null,\ + \"right\" : null,\ \"value\" : {\ \"name\" : \"Peter\", \ \"age\" : 55 \ }\ - }\ + },\ + \"value\" : null\ }\ }"; @@ -222,6 +220,8 @@ static void check_example5(void *data) { ASSERT_EQ(55, ex->head->right->value->age); ASSERT_TRUE(ex->head->right->left == nullptr); ASSERT_TRUE(ex->head->right->right == nullptr); + + ASSERT_TRUE(ex->head->value == nullptr); } /*********** example 6 ************************/ @@ -280,25 +280,6 @@ const char *example8_input = "{ \ \"e\" : 32 \ }"; -struct example8 { - bool a; - unsigned char b; - uint32_t c; - uint64_t d; - int e; - void* f; -}; - -static void check_example8(void *data) { - auto ex = static_cast(data); - ASSERT_EQ(true,ex->a); - ASSERT_EQ(4,ex->b); - ASSERT_EQ(8,ex->c); - //error on mac ASSERT_EQ(16,ex->d); - ASSERT_TRUE(16 == ex->d); - ASSERT_EQ(32,ex->e); -} - /*********** example 9 ************************/ const char *example9_descriptor = "{It#OK=0;#NOK=1;#MAYBE=2;E id name result}"; @@ -429,7 +410,7 @@ static void parseTests() { ASSERT_EQ(0, rc); rc = jsonSerializer_deserialize(type, example5_input, strlen(example5_input), &inst); ASSERT_EQ(0, rc); - check_example5(inst); + check_example5(inst); dynType_free(type, inst); dynType_destroy(type); @@ -453,14 +434,15 @@ static void parseTests() { dynType_free(type, inst); dynType_destroy(type); + // missing object member "f" type = nullptr; inst = nullptr; rc = dynType_parseWithStr(example8_descriptor, nullptr, nullptr, &type); ASSERT_EQ(0, rc); rc = jsonSerializer_deserialize(type, example8_input, strlen(example8_input), &inst); - ASSERT_EQ(0, rc); - check_example8(inst); - dynType_free(type, inst); + ASSERT_NE(0, rc); + EXPECT_STREQ("Error cannot deserialize json. Input is '{ \"a\" : true, \"b\" : 4, \"c\" : 8, \"d\" : 16, \"e\" : 32 }'", + celix_err_popLastError()); dynType_destroy(type); type = nullptr; @@ -516,12 +498,14 @@ static void parseTests() { celix_err_printErrors(stderr, nullptr, nullptr); dynType_destroy(type); - //pointer type mismatch + //double pointer unsupported rc = dynType_parseWithStr("{*t a}", nullptr, nullptr, &type); ASSERT_EQ(0, rc); - inputStr = R"({"a":1.0})"; + inputStr = R"({"a":"hello world"})"; rc = jsonSerializer_deserialize(type, inputStr, strlen(inputStr), &inst); ASSERT_EQ(1, rc); + EXPECT_STREQ("Error cannot deserialize json. Input is '{\"a\":\"hello world\"}'", celix_err_popLastError()); + EXPECT_STREQ("Error cannot deserialize pointer to pointer", celix_err_popLastError()); celix_err_printErrors(stderr, nullptr, nullptr); dynType_destroy(type); @@ -569,6 +553,76 @@ static void parseTests() { ASSERT_EQ(1, rc); celix_err_printErrors(stderr, nullptr, nullptr); dynType_destroy(type); + + // extra member ("b") is allowed + rc = dynType_parseWithStr("{t a}", nullptr, nullptr, &type); + ASSERT_EQ(0, rc); + inputStr = R"({"a":"hello", "b":"world"})"; + rc = jsonSerializer_deserialize(type, inputStr, strlen(inputStr), &inst); + ASSERT_EQ(0, rc); + dynType_free(type, inst); + dynType_destroy(type); + + // parse complex from non-object + rc = dynType_parseWithStr("{t a}", nullptr, nullptr, &type); + ASSERT_EQ(0, rc); + inputStr = R"(["a"])"; + rc = jsonSerializer_deserialize(type, inputStr, strlen(inputStr), &inst); + ASSERT_EQ(1, rc); + celix_err_printErrors(stderr, nullptr, nullptr); + dynType_destroy(type); + + //simple string + rc = dynType_parseWithStr("t", nullptr, nullptr, &type); + ASSERT_EQ(0, rc); + inputStr = R"("hello")"; + rc = jsonSerializer_deserialize(type, inputStr, strlen(inputStr), &inst); + ASSERT_EQ(0, rc); + EXPECT_STREQ("hello", *(char**)inst); + dynType_free(type, inst); + dynType_destroy(type); + + //null string + rc = dynType_parseWithStr("t", nullptr, nullptr, &type); + ASSERT_EQ(0, rc); + inputStr = R"(null)"; + rc = jsonSerializer_deserialize(type, inputStr, strlen(inputStr), &inst); + ASSERT_EQ(0, rc); + EXPECT_EQ(nullptr, *(char**)inst); + dynType_free(type, inst); + dynType_destroy(type); + + // double* + rc = dynType_parseWithStr("*D", nullptr, nullptr, &type); + ASSERT_EQ(0, rc); + inputStr = R"(0.1)"; + rc = jsonSerializer_deserialize(type, inputStr, strlen(inputStr), &inst); + ASSERT_EQ(0, rc); + EXPECT_EQ(0.1, **(double**)inst); + dynType_free(type, inst); + dynType_destroy(type); + + // null double* + rc = dynType_parseWithStr("*D", nullptr, nullptr, &type); + ASSERT_EQ(0, rc); + inputStr = R"(null)"; + rc = jsonSerializer_deserialize(type, inputStr, strlen(inputStr), &inst); + ASSERT_EQ(0, rc); + EXPECT_EQ(nullptr, *(double**)inst); + dynType_free(type, inst); + dynType_destroy(type); + + // unnamed fields + type = nullptr; + inst = nullptr; + rc = dynType_parseWithStr("{DD}", nullptr, nullptr, &type); + ASSERT_EQ(0, rc); + inputStr = R"({"a":1.0, "b":2.0})"; + rc = jsonSerializer_deserialize(type, inputStr, strlen(inputStr), &inst); + EXPECT_NE(0, rc); + EXPECT_STREQ("Error cannot deserialize json. Input is '{\"a\":1.0, \"b\":2.0}'", celix_err_popLastError()); + EXPECT_STREQ("Unamed field unsupported", celix_err_popLastError()); + dynType_destroy(type); } /*********** write example 1 ************************/ @@ -780,13 +834,9 @@ void writeEnumFailed(void) { class JsonSerializerTests : public ::testing::Test { public: JsonSerializerTests() { - int lvl = 1; - dynCommon_logSetup(stdLog, nullptr, lvl); - dynType_logSetup(stdLog, nullptr,lvl); - dynTypeCommon_logSetup(stdLog, nullptr,lvl); - jsonSerializer_logSetup(stdLog, nullptr, lvl); } ~JsonSerializerTests() override { + celix_err_resetErrors(); } }; @@ -819,10 +869,120 @@ TEST_F(JsonSerializerTests, WriteSequenceFailed) { writeSequenceFailed(); } +TEST_F(JsonSerializerTests, WriteSequenceFailed2) { + dyn_type* type = nullptr; + char* result = nullptr; + int rc = dynType_parseWithStr(R"([**D)", nullptr, nullptr, &type); + ASSERT_EQ(0, rc); + double** input[] = {nullptr}; + struct { + uint32_t cap; + uint32_t len; + double*** buf; + }seq{1,1,input}; + rc = jsonSerializer_serialize(type, &seq, &result); + ASSERT_EQ(1, rc); + celix_err_printErrors(stderr, nullptr, nullptr); + dynType_destroy(type); +} + TEST_F(JsonSerializerTests, WriteComplexFailed) { writeComplexFailed(); } +TEST_F(JsonSerializerTests, WriteComplexFailed2) { + dyn_type *type = nullptr; + char *result = nullptr; + int rc = dynType_parseWithStr(R"({**D a})", nullptr, nullptr, &type); + ASSERT_EQ(0, rc); + struct { + double** a; + }input{nullptr}; + rc = jsonSerializer_serialize(type, &input, &result); + ASSERT_EQ(1, rc); + celix_err_printErrors(stderr, nullptr, nullptr); + dynType_destroy(type); +} + TEST_F(JsonSerializerTests, WriteEnumFailed) { writeEnumFailed(); } + +TEST_F(JsonSerializerTests, WriteDoublePointer) { + dyn_type *type; + int rc; + double val = 1.0; + double* valp = &val; + double** valpp = &valp; + char *result = nullptr; + + rc = dynType_parseWithStr("**D", nullptr, nullptr, &type); + ASSERT_EQ(0, rc); + rc = jsonSerializer_serialize(type, &valpp, &result); + EXPECT_NE(0, rc); + EXPECT_STREQ("Error cannot serialize pointer to pointer", celix_err_popLastError()); + dynType_destroy(type); +} + +TEST_F(JsonSerializerTests, SerializationDeserilizationTest) { + dyn_type *type; + void *inst; + int rc; + + type = nullptr; + inst = nullptr; + rc = dynType_parseWithStr(example5_descriptor, nullptr, nullptr, &type); + ASSERT_EQ(0, rc); + json_auto_t* root = json_loads(example5_input, JSON_DECODE_ANY, NULL); + ASSERT_NE(nullptr, root); + rc = jsonSerializer_deserializeJson(type, root, &inst); + ASSERT_EQ(0, rc); + check_example5(inst); + + json_auto_t* result = nullptr; + rc = jsonSerializer_serializeJson(type, inst, &result); + ASSERT_EQ(0, rc); + EXPECT_TRUE(json_equal(root, result)); + dynType_free(type, inst); + dynType_destroy(type); +} + +TEST_F(JsonSerializerTests, SerializationDeserilizationNullStringTest) { + dyn_type *type; + void *inst; + int rc; + rc = dynType_parseWithStr("t", nullptr, nullptr, &type); + ASSERT_EQ(0, rc); + json_auto_t* root = json_null(); + rc = jsonSerializer_deserializeJson(type, root, &inst); + ASSERT_EQ(0, rc); + EXPECT_EQ(nullptr, *(char**)inst); + + json_auto_t* result = nullptr; + rc = jsonSerializer_serializeJson(type, inst, &result); + ASSERT_EQ(0, rc); + EXPECT_TRUE(json_equal(root, result)); + dynType_free(type, inst); + dynType_destroy(type); + +} + +TEST_F(JsonSerializerTests, SerializationDeserilizationStringTest) { + dyn_type *type; + void *inst; + int rc; + rc = dynType_parseWithStr("t", nullptr, nullptr, &type); + ASSERT_EQ(0, rc); + json_auto_t* root = json_loads(R"("hello")", JSON_DECODE_ANY, NULL); + ASSERT_NE(nullptr, root); + rc = jsonSerializer_deserializeJson(type, root, &inst); + ASSERT_EQ(0, rc); + EXPECT_STREQ("hello", *(char**)inst); + + json_auto_t* result = nullptr; + rc = jsonSerializer_serializeJson(type, inst, &result); + ASSERT_EQ(0, rc); + EXPECT_TRUE(json_equal(root, result)); + dynType_free(type, inst); + dynType_destroy(type); +} diff --git a/libs/dfi/include/dfi_log_util.h b/libs/dfi/include/dfi_log_util.h deleted file mode 100644 index 9c65d6b17..000000000 --- a/libs/dfi/include/dfi_log_util.h +++ /dev/null @@ -1,78 +0,0 @@ -/* - * 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. - */ - -#ifndef _DFI_LOG_UTIL_H_ -#define _DFI_LOG_UTIL_H_ - -/** - * @deprecated The header will be remove, Use celix_err.h instead. - */ - -#include "celix_dfi_export.h" -#include "celix_err.h" - -#ifdef __cplusplus -extern "C" { -#endif - -typedef void (*logf_ft)(void *handle, int level, const char *file, int line, const char *format, ...); - -#define DFI_SETUP_LOG_HEADER(cmp) \ - void cmp ## _logSetup(logf_ft logf, void *handle, int currentLogLevel); - -#define DFI_SETUP_LOG(cmp) \ - static logf_ft g_logf = NULL; \ - static void *g_logHandle = NULL; \ - static int g_currentLogLevel = 1; \ - \ - CELIX_DFI_EXPORT void cmp ## _logSetup(logf_ft logf, void *handle, int currentLogLevel) { \ - g_currentLogLevel = currentLogLevel; \ - g_logHandle = handle; \ - g_logf = logf; \ - } - -#define LOG_LVL_ERROR 1 -#define LOG_LVL_WARNING 2 -#define LOG_LVL_INFO 3 -#define LOG_LVL_DEBUG 4 - -#define LOG_ERROR(msg, ...) celix_err_pushf((msg), ##__VA_ARGS__) - -#define LOG_WARNING(msg, ...) \ - if (g_logf != NULL && g_currentLogLevel >= LOG_LVL_WARNING) { \ - g_logf(g_logHandle, LOG_LVL_WARNING, __FILE__, __LINE__, (msg), ##__VA_ARGS__); \ - } - -#define LOG_INFO(msg, ...) \ - if (g_logf != NULL && g_currentLogLevel >= LOG_LVL_INFO) { \ - g_logf(g_logHandle, LOG_LVL_INFO, __FILE__, __LINE__, (msg), ##__VA_ARGS__); \ - } - -#define LOG_DEBUG(msg, ...) \ - if (g_logf != NULL && g_currentLogLevel >= LOG_LVL_DEBUG) { \ - g_logf(g_logHandle, LOG_LVL_DEBUG, __FILE__, __LINE__, (msg), ##__VA_ARGS__); \ - } - - -#ifdef __cplusplus -} -#endif - - -#endif diff --git a/libs/dfi/include/dyn_function.h b/libs/dfi/include/dyn_function.h index e812eeaf8..b548fdb1e 100644 --- a/libs/dfi/include/dyn_function.h +++ b/libs/dfi/include/dyn_function.h @@ -21,7 +21,7 @@ #define __DYN_FUNCTION_H_ #include "dyn_type.h" -#include "dfi_log_util.h" +#include "celix_cleanup.h" #include "celix_dfi_export.h" #ifdef __cplusplus @@ -34,8 +34,10 @@ extern "C" { * * Dyn function argument meta (am) as meta info, with the following possible values * am=handle #void pointer for the handle - * am=pre #output pointer with memory pre-allocated - * am=out #output pointer + * am=pre #output pointer with memory pre-allocated, it should be pointer to trivial types, check `dynType_isTrivial` for more info. + * am=out #output pointer, it should be pointers to text or pointer to serializable point type + * + * Without meta info the argument is considered to be a standard argument, which can be of any serializable type. * * text argument (t) can also be annotated to be considered const string. * Normally a text argument will be handled as char*, meaning that the callee is expected to take of ownership. @@ -45,8 +47,6 @@ extern "C" { typedef struct _dyn_function_type dyn_function_type; -DFI_SETUP_LOG_HEADER(dynFunction); - enum dyn_function_argument_meta { DYN_FUNCTION_ARGUMENT_META__STD = 0, DYN_FUNCTION_ARGUMENT_META__HANDLE = 1, @@ -54,6 +54,15 @@ enum dyn_function_argument_meta { DYN_FUNCTION_ARGUMENT_META__OUTPUT = 3 }; +typedef struct _dyn_function_argument_type dyn_function_argument_type; +TAILQ_HEAD(dyn_function_arguments_head,_dyn_function_argument_type); +struct _dyn_function_argument_type { + int index; + enum dyn_function_argument_meta argumentMeta; + dyn_type* type; + TAILQ_ENTRY(_dyn_function_argument_type) entries; +}; + /** * @brief Creates a dyn_function_type according to the given function descriptor stream. * @@ -68,7 +77,7 @@ enum dyn_function_argument_meta { * @retval 1 If there is not enough memory to create dyn_function_type. * @retval 2 Errors other than out-of-memory. */ -CELIX_DFI_EXPORT int dynFunction_parse(FILE *descriptorStream, struct types_head *refTypes, dyn_function_type **dynFunc); +CELIX_DFI_EXPORT int dynFunction_parse(FILE* descriptorStream, struct types_head* refTypes, dyn_function_type** dynFunc); /** * @brief Creates a dyn_function_type according to the given function descriptor string. @@ -84,14 +93,14 @@ CELIX_DFI_EXPORT int dynFunction_parse(FILE *descriptorStream, struct types_head * @retval 1 If there is not enough memory to create dyn_function_type. * @retval 2 Errors other than out-of-memory. */ -CELIX_DFI_EXPORT int dynFunction_parseWithStr(const char *descriptor, struct types_head *refTypes, dyn_function_type **dynFunc); +CELIX_DFI_EXPORT int dynFunction_parseWithStr(const char* descriptor, struct types_head* refTypes, dyn_function_type** dynFunc); /** * @brief Returns the number of arguments of the given dynamic function type instance. * @param[in] dynFunc The dynamic type instance for function. * @return The number of arguments. */ -CELIX_DFI_EXPORT int dynFunction_nrOfArguments(dyn_function_type *dynFunc); +CELIX_DFI_EXPORT int dynFunction_nrOfArguments(const dyn_function_type* dynFunc); /** * @brief Returns the argument type for the given argument index. @@ -99,7 +108,7 @@ CELIX_DFI_EXPORT int dynFunction_nrOfArguments(dyn_function_type *dynFunc); * @param[in] argumentNr The argument index. * @return The argument type. */ -CELIX_DFI_EXPORT dyn_type *dynFunction_argumentTypeForIndex(dyn_function_type *dynFunc, int argumentNr); +CELIX_DFI_EXPORT const dyn_type* dynFunction_argumentTypeForIndex(const dyn_function_type* dynFunc, int argumentNr); /** * @brief Returns the argument meta for the given argument index. @@ -107,20 +116,28 @@ CELIX_DFI_EXPORT dyn_type *dynFunction_argumentTypeForIndex(dyn_function_type *d * @param[in] argumentNr The argument index. * @return The argument meta. */ -CELIX_DFI_EXPORT enum dyn_function_argument_meta dynFunction_argumentMetaForIndex(dyn_function_type *dynFunc, int argumentNr); +CELIX_DFI_EXPORT enum dyn_function_argument_meta dynFunction_argumentMetaForIndex(const dyn_function_type* dynFunc, int argumentNr); + +/** + * @brief Returns the argument list for the given dynamic function type instance. + * @note It always returns valid list. + */ +CELIX_DFI_EXPORT const struct dyn_function_arguments_head* dynFunction_arguments(const dyn_function_type* dynFunc); /** * @brief Returns the return value type for the given dynamic function type instance. * @param[in] dynFunc The dynamic type instance for function. * @return The return value type. */ -CELIX_DFI_EXPORT dyn_type * dynFunction_returnType(dyn_function_type *dynFunction); +CELIX_DFI_EXPORT const dyn_type* dynFunction_returnType(const dyn_function_type* dynFunction); /** * @brief Destroys the given dynamic function type instance. * @param[in] dynFunc The dynamic type instance for function. */ -CELIX_DFI_EXPORT void dynFunction_destroy(dyn_function_type *dynFunc); +CELIX_DFI_EXPORT void dynFunction_destroy(dyn_function_type* dynFunc); + +CELIX_DEFINE_AUTOPTR_CLEANUP_FUNC(dyn_function_type, dynFunction_destroy); /** * @brief Calls the given dynamic type function. @@ -130,7 +147,7 @@ CELIX_DFI_EXPORT void dynFunction_destroy(dyn_function_type *dynFunc); * @param[in] argValues The argument values. * @return 0 */ -CELIX_DFI_EXPORT int dynFunction_call(dyn_function_type *dynFunc, void(*fn)(void), void *returnValue, void **argValues); +CELIX_DFI_EXPORT int dynFunction_call(const dyn_function_type* dynFunc, void(*fn)(void), void* returnValue, void** argValues); /** * @brief Creates a closure for the given dynamic function type instance. @@ -142,7 +159,7 @@ CELIX_DFI_EXPORT int dynFunction_call(dyn_function_type *dynFunc, void(*fn)(void * @retval 1 If there is not enough memory to create the closure. * @retval 2 Errors other than out-of-memory. */ -CELIX_DFI_EXPORT int dynFunction_createClosure(dyn_function_type *func, void (*bind)(void *, void **, void*), void *userData, void(**fn)(void)); +CELIX_DFI_EXPORT int dynFunction_createClosure(dyn_function_type* func, void (*bind)(void*, void**, void*), void* userData, void(**fn)(void)); /** * @brief Returns the function pointer for the given dynamic function type instance. @@ -150,13 +167,18 @@ CELIX_DFI_EXPORT int dynFunction_createClosure(dyn_function_type *func, void (*b * @param[out] fn The function pointer. * @return 0 If successful, 1 if the dynamic function type instance has no function pointer. */ -CELIX_DFI_EXPORT int dynFunction_getFnPointer(dyn_function_type *func, void (**fn)(void)); +CELIX_DFI_EXPORT int dynFunction_getFnPointer(const dyn_function_type* func, void (**fn)(void)); /** * Returns whether the function has a return type. * Will return false if return is void. */ -CELIX_DFI_EXPORT bool dynFunction_hasReturn(dyn_function_type *dynFunction); +CELIX_DFI_EXPORT bool dynFunction_hasReturn(const dyn_function_type* dynFunction); + +/** + * @brief Returns the name of the given dynamic function type instance. + */ +CELIX_DFI_EXPORT const char* dynFunction_getName(const dyn_function_type* func); #ifdef __cplusplus } diff --git a/libs/dfi/include/dyn_interface.h b/libs/dfi/include/dyn_interface.h index 0f80bec23..1694dbda3 100644 --- a/libs/dfi/include/dyn_interface.h +++ b/libs/dfi/include/dyn_interface.h @@ -20,10 +20,8 @@ #ifndef __DYN_INTERFACE_H_ #define __DYN_INTERFACE_H_ -#include "dyn_common.h" #include "dyn_type.h" #include "dyn_function.h" -#include "dfi_log_util.h" #include "celix_cleanup.h" #include "celix_version.h" #include "celix_dfi_export.h" @@ -32,8 +30,6 @@ extern "C" { #endif -DFI_SETUP_LOG_HEADER(dynInterface); - /* Description string * * Descriptor (interface) = HeaderSection AnnotationSection TypesSection MethodsSection @@ -49,10 +45,9 @@ typedef struct _dyn_interface_type dyn_interface_type; TAILQ_HEAD(methods_head, method_entry); struct method_entry { - int index; - char *id; - char *name; - dyn_function_type *dynFunc; + int index; ///< The index of the method in the interface + char* id; ///< The signature of the method + dyn_function_type* dynFunc; ///< The function type of the method TAILQ_ENTRY(method_entry) entries; }; @@ -68,52 +63,36 @@ struct method_entry { * @param[out] out The created dynamic interface type instance. * @return 0 if successful, 1 otherwise. */ -CELIX_DFI_EXPORT int dynInterface_parse(FILE *descriptor, dyn_interface_type **out); +CELIX_DFI_EXPORT int dynInterface_parse(FILE* descriptor, dyn_interface_type** out); /** * @brief Destroys the given dynamic interface type instance. * @param[in] intf The dynamic interface type instance to destroy. */ -CELIX_DFI_EXPORT void dynInterface_destroy(dyn_interface_type *intf); +CELIX_DFI_EXPORT void dynInterface_destroy(dyn_interface_type* intf); CELIX_DEFINE_AUTOPTR_CLEANUP_FUNC(dyn_interface_type, dynInterface_destroy); /** * @brief Gets the name of the given dynamic interface type instance. - * * The dynamic interface type instance is the owner of the returned string and the string should not be freed. - * - * In case of an error, an error message is added to celix_err. - * - * @param[in] intf The dynamic interface type instance. - * @param[out] name The name of the dynamic interface type instance. - * @return 0 if successful, 1 otherwise. + * @note It always returns valid string. */ -CELIX_DFI_EXPORT int dynInterface_getName(dyn_interface_type *intf, char **name); +CELIX_DFI_EXPORT const char* dynInterface_getName(const dyn_interface_type* intf); /** * @brief Gets the version of the given dynamic interface type instance. - * * The dynamic interface type instance is the owner of the version and the version should not be freed. - * - * @param[in] intf The dynamic interface type instance. - * @param[out] version The version of the dynamic interface type instance. - * @return 0 if successful, 1 otherwise. + * @note It always returns valid version. */ -CELIX_DFI_EXPORT int dynInterface_getVersion(dyn_interface_type *intf, celix_version_t** version); +CELIX_DFI_EXPORT const celix_version_t* dynInterface_getVersion(const dyn_interface_type* intf); /** * @brief Gets the version string of the given dynamic interface type instance. - * * The dynamic interface type instance is the owner of the version string and the version string should not be freed. - * - * In case of an error, an error message is added to celix_err. - * - * @param[in] intf The dynamic interface type instance. - * @param[out] version The version string of the dynamic interface type instance. - * @return 0 if successful, 1 otherwise. + * @note It always returns valid string. */ -CELIX_DFI_EXPORT int dynInterface_getVersionString(dyn_interface_type *intf, char **version); +CELIX_DFI_EXPORT const char* dynInterface_getVersionString(const dyn_interface_type* intf); /** * @brief Gets the value corresponding to the specified name, which comes from the header section of the given dynamic interface type instance. @@ -127,7 +106,7 @@ CELIX_DFI_EXPORT int dynInterface_getVersionString(dyn_interface_type *intf, cha * @param[out] value The value corresponding to the specified name. * @return 0 if successful, 1 otherwise. */ -CELIX_DFI_EXPORT int dynInterface_getHeaderEntry(dyn_interface_type *intf, const char *name, char **value); +CELIX_DFI_EXPORT int dynInterface_getHeaderEntry(const dyn_interface_type* intf, const char* name, const char** value); /** * @brief Gets the value corresponding to the specified name, which comes from the annotation section of the given dynamic interface type instance. @@ -141,25 +120,31 @@ CELIX_DFI_EXPORT int dynInterface_getHeaderEntry(dyn_interface_type *intf, const * @param[out] value The value corresponding to the specified name. * @return 0 if successful, 1 otherwise. */ -CELIX_DFI_EXPORT int dynInterface_getAnnotationEntry(dyn_interface_type *intf, const char *name, char **value); +CELIX_DFI_EXPORT int dynInterface_getAnnotationEntry(const dyn_interface_type* intf, const char* name, const char** value); /** * @brief Gets the methods of the given dynamic interface type instance. - * * The dynamic interface type instance is the owner of the list and the list should not be freed. - * - * @param[in] intf The dynamic interface type instance. - * @param[out] list The method list of the dynamic interface type instance. - * @return 0. + * @note It always returns valid methods. */ -CELIX_DFI_EXPORT int dynInterface_methods(dyn_interface_type *intf, struct methods_head **list); +CELIX_DFI_EXPORT const struct methods_head* dynInterface_methods(const dyn_interface_type* intf); /** * @brief Returns the number of methods for the given dynamic interface type instance. * @param[in] intf The dynamic interface type instance. * @return The number of methods for the given dynamic interface type instance. */ -CELIX_DFI_EXPORT int dynInterface_nrOfMethods(dyn_interface_type *intf); +CELIX_DFI_EXPORT int dynInterface_nrOfMethods(const dyn_interface_type* intf); + +/** + * @brief Finds and returns the method_entry structure for a given method id in the dynamic interface type instance. + * The dynamic interface type instance is the owner of the returned method_entry structure and it should not be freed. + * + * @param[in] intf The dynamic interface type instance. + * @param[in] id The id of the method to find, which is currently the signature of the method. + * @return The method_entry structure for the given method id, or NULL if no matching method is found. + */ +CELIX_DFI_EXPORT const struct method_entry* dynInterface_findMethod(const dyn_interface_type* intf, const char* id); #ifdef __cplusplus diff --git a/libs/dfi/include/dyn_message.h b/libs/dfi/include/dyn_message.h index 2b719efae..3564ce674 100644 --- a/libs/dfi/include/dyn_message.h +++ b/libs/dfi/include/dyn_message.h @@ -20,9 +20,8 @@ #ifndef __DYN_MESSAGE_H_ #define __DYN_MESSAGE_H_ -#include "dyn_common.h" #include "dyn_type.h" -#include "dfi_log_util.h" +#include "celix_cleanup.h" #include "celix_version.h" #include "celix_dfi_export.h" @@ -30,8 +29,6 @@ extern "C" { #endif -DFI_SETUP_LOG_HEADER(dynMessage); - /* Description string * * Descriptor (message) = HeaderSection AnnotationSection TypesSection MessageSection @@ -57,50 +54,36 @@ typedef struct _dyn_message_type dyn_message_type; * @param[out] out The dynamic message type instance. * @return 0 if successful, 1 otherwise. */ -CELIX_DFI_EXPORT int dynMessage_parse(FILE *descriptor, dyn_message_type **out); +CELIX_DFI_EXPORT int dynMessage_parse(FILE* descriptor, dyn_message_type** out); /** * @brief Destroy the dynamic message type instance. * @param[in] msg The dynamic message type instance. */ -CELIX_DFI_EXPORT void dynMessage_destroy(dyn_message_type *msg); +CELIX_DFI_EXPORT void dynMessage_destroy(dyn_message_type* msg); + +CELIX_DEFINE_AUTOPTR_CLEANUP_FUNC(dyn_message_type, dynMessage_destroy); /** * @brief Gets the name of the dynamic message type instance. - * * The dynamic message type instance is the owner of the name and the name should not be freed. - * - * In case of an error, an error message is added to celix_err. - * - * @param[in] msg The dynamic message type instance. - * @param[out] name The name of the dynamic message type instance. - * @return 0 if successful, 1 otherwise. + * @note It always returns valid string. */ -CELIX_DFI_EXPORT int dynMessage_getName(dyn_message_type *msg, char **name); +CELIX_DFI_EXPORT const char* dynMessage_getName(const dyn_message_type* msg); /** * @brief Gets the version of the given dynamic message type instance. - * * The dynamic message type instance is the owner of the version and the version should not be freed. - * - * @param[in] msg The dynamic message type instance. - * @param[out] version The version of the dynamic message type instance. - * @return 0 if successful, 1 otherwise. + * @note It always returns valid version. */ -CELIX_DFI_EXPORT int dynMessage_getVersion(dyn_message_type *msg, celix_version_t** version); +CELIX_DFI_EXPORT const celix_version_t* dynMessage_getVersion(const dyn_message_type* msg); /** * @brief Gets the version string of the given dynamic message type instance. - * * The dynamic message type instance is the owner of the version string and the version string should not be freed. - * - * In case of an error, an error message is added to celix_err. - * - * @param[in] msg The dynamic message type instance. - * @param[out] version The version string of the dynamic message type instance. - * @return 0 if successful, 1 otherwise. + * @note It always returns valid string. */ -CELIX_DFI_EXPORT int dynMessage_getVersionString(dyn_message_type *msg, char **version); +CELIX_DFI_EXPORT const char* dynMessage_getVersionString(const dyn_message_type* msg); /** * @brief Gets the value corresponding to the specified name, which comes from the header section of the given dynamic message type instance. @@ -114,7 +97,7 @@ CELIX_DFI_EXPORT int dynMessage_getVersionString(dyn_message_type *msg, char **v * @param[out] value The value corresponding to the specified name. * @return 0 if successful, 1 otherwise. */ -CELIX_DFI_EXPORT int dynMessage_getHeaderEntry(dyn_message_type *msg, const char *name, char **value); +CELIX_DFI_EXPORT int dynMessage_getHeaderEntry(dyn_message_type* msg, const char* name, const char** value); /** * @brief Gets the value corresponding to the specified name, which comes from the annotation section of the given dynamic message type instance. @@ -128,18 +111,14 @@ CELIX_DFI_EXPORT int dynMessage_getHeaderEntry(dyn_message_type *msg, const char * @param[out] value The value corresponding to the specified name. * @return 0 if successful, 1 otherwise. */ -CELIX_DFI_EXPORT int dynMessage_getAnnotationEntry(dyn_message_type *msg, const char *name, char **value); +CELIX_DFI_EXPORT int dynMessage_getAnnotationEntry(dyn_message_type* msg, const char* name, const char** value); /** * @brief Gets the message type from the given dynamic message type instance. - * * The dynamic message type instance is the owner of the message type and the message type should not be freed. - * - * @param[in] msg The dynamic message type instance. - * @param[out] type The message type. - * @return 0 + * @note It always returns valid dyn_type. */ -CELIX_DFI_EXPORT int dynMessage_getMessageType(dyn_message_type *msg, dyn_type **type); +CELIX_DFI_EXPORT const dyn_type* dynMessage_getMessageType(dyn_message_type* msg); #ifdef __cplusplus } diff --git a/libs/dfi/include/dyn_type.h b/libs/dfi/include/dyn_type.h index c8a963eaf..77f4dbec7 100644 --- a/libs/dfi/include/dyn_type.h +++ b/libs/dfi/include/dyn_type.h @@ -25,18 +25,13 @@ #include #include -#include "dfi_log_util.h" #include "celix_dfi_export.h" +#include "celix_cleanup.h" #ifdef __cplusplus extern "C" { #endif -#if defined(NO_MEMSTREAM_AVAILABLE) -#include "memstream/open_memstream.h" -#include "memstream/fmemopen.h" -#endif - /** * dyn type (dynamic type) represent a structure in memory. It can be used to calculate the needed memory size, the size * and offset of struct members and can be used to dynamically alloc, initialize, dealloc, read & write structure @@ -64,8 +59,8 @@ extern "C" { * //Extended * b unsigned char * i uint32_t - * j uint62_t - * s uint64_t + * j uint64_t + * s uint16_t * P untyped pointer (void *) * t char* string * N native int @@ -117,65 +112,70 @@ typedef struct _dyn_type dyn_type; TAILQ_HEAD(types_head, type_entry); struct type_entry { - dyn_type *type; + dyn_type* type; TAILQ_ENTRY(type_entry) entries; }; TAILQ_HEAD(complex_type_entries_head, complex_type_entry); struct complex_type_entry { - dyn_type *type; - char *name; + dyn_type* type; + char* name; TAILQ_ENTRY(complex_type_entry) entries; }; TAILQ_HEAD(meta_properties_head, meta_entry); struct meta_entry { - char *name; - char *value; + char* name; + char* value; TAILQ_ENTRY(meta_entry) entries; }; -//logging -DFI_SETUP_LOG_HEADER(dynType); - /** * Parses a descriptor stream and creates a dyn_type (dynamic type). * If successful the type output argument points to the newly created dyn type. - * The caller is the owner of the dyn type and use dynType_destroy deallocate the memory. + * The caller is the owner of the dyn type and use dynType_destroy to deallocate the memory. * * In case of an error, an error message is added to celix_err. * * @param descriptorStream Stream to the descriptor. * @param name name for the dyn_type. This can be used in references to dyn types. - * @param refTypes A list if reference-able dyn types. + * @param refTypes A list of referable dyn types. * @param type Output argument for the parsed dyn type. * @return 0 if successful. */ -CELIX_DFI_EXPORT int dynType_parse(FILE *descriptorStream, const char *name, struct types_head *refTypes, dyn_type **type); +CELIX_DFI_EXPORT int dynType_parse(FILE* descriptorStream, const char* name, const struct types_head* refTypes, dyn_type** type); + +/** + * @brief Parses a descriptor stream and creates a dyn_type (dynamic type) of a given name. + * Similar to dynType_parse except that the ownership of the given name is taken by the dyn type. + */ +CELIX_DFI_EXPORT int dynType_parseOfName(FILE* descriptorStream, char* name, const struct types_head* refTypes, dyn_type** type); /** * Parses a descriptor string and creates a dyn_type (dynamic type). * If successful the type output argument points to the newly created dyn type. - * The caller is the owner of the dyn type and use dynType_destroy deallocate the memory. + * The caller is the owner of the dyn type and use dynType_destroy to deallocate the memory. * * In case of an error, an error message is added to celix_err. * * @param descriptor The descriptor. * @param name name for the dyn_type. This can be used in references to dyn types. - * @param refTypes A list if reference-able dyn types. + * @param refTypes A list if referable dyn types. * @param type Output argument for the parsed dyn type. * @return 0 if successful. */ -CELIX_DFI_EXPORT int dynType_parseWithStr(const char *descriptor, const char *name, struct types_head *refTypes, dyn_type **type); +CELIX_DFI_EXPORT int dynType_parseWithStr(const char* descriptor, const char* name, const struct types_head* refTypes, dyn_type** type); /** * Destroy a dyn type and de-allocates the memory. * @param type The dyn type to destroy. */ -CELIX_DFI_EXPORT void dynType_destroy(dyn_type *type); +CELIX_DFI_EXPORT void dynType_destroy(dyn_type* type); + +CELIX_DEFINE_AUTOPTR_CLEANUP_FUNC(dyn_type, dynType_destroy); /** - * Allocates memory for a type instance described by a dyn type. The memory will be 0 allocated (calloc). + * Allocates memory for a type instance described by a dyn type. The memory will be set to zero (calloc). * * In case of an error, an error message is added to celix_err. * @@ -183,7 +183,7 @@ CELIX_DFI_EXPORT void dynType_destroy(dyn_type *type); * @param instance The output argument for the allocated memory. * @return 0 on success. */ -CELIX_DFI_EXPORT int dynType_alloc(dyn_type *type, void **instance); +CELIX_DFI_EXPORT int dynType_alloc(const dyn_type* type, void** instance); /** * free the memory for a type instance described by a dyn type. @@ -192,23 +192,23 @@ CELIX_DFI_EXPORT int dynType_alloc(dyn_type *type, void **instance); * @param type The dyn type for which structure to allocate. * @param instance The memory location of the type instance. */ -CELIX_DFI_EXPORT void dynType_free(dyn_type *type, void *instance); +CELIX_DFI_EXPORT void dynType_free(const dyn_type* type, void* instance); /** * Prints the dyn type information to the provided output stream. * @param type The dyn type to print. * @param stream The output stream (e.g. stdout). */ -CELIX_DFI_EXPORT void dynType_print(dyn_type *type, FILE *stream); +CELIX_DFI_EXPORT void dynType_print(const dyn_type* type, FILE* stream); /** - * Return the size of the structure descripbed by the dyn type. - * This this _not_ includes sizes of data where is only pointed to (i.e. content of sequences, strings, etc). + * Return the size of the structure described by the dyn type. + * This does _not_ includes sizes of data which is only pointed to (i.e. content of sequences, strings, etc). * * @param type The dyn type. * @return The size of the type instance described by dyn type. */ -CELIX_DFI_EXPORT size_t dynType_size(dyn_type *type); +CELIX_DFI_EXPORT size_t dynType_size(const dyn_type* type); /** * The type of the dyn type @@ -216,7 +216,12 @@ CELIX_DFI_EXPORT size_t dynType_size(dyn_type *type); * @param type The dyn type * @return The type of the dyn type. */ -CELIX_DFI_EXPORT int dynType_type(dyn_type *type); +CELIX_DFI_EXPORT int dynType_type(const dyn_type* type); + +/** + * Returns the real (non-reference) type of the dyn type. + */ +CELIX_DFI_EXPORT const dyn_type* dynType_realType(const dyn_type* type); /** * Returns the char identifier of the dyn type type. @@ -224,7 +229,7 @@ CELIX_DFI_EXPORT int dynType_type(dyn_type *type); * @param type The dyn type * @return The descriptor of the dyn type. */ -CELIX_DFI_EXPORT char dynType_descriptorType(dyn_type *type); +CELIX_DFI_EXPORT char dynType_descriptorType(const dyn_type* type); /** * Get the dyn type meta information for the provided name. @@ -232,21 +237,20 @@ CELIX_DFI_EXPORT char dynType_descriptorType(dyn_type *type); * @param name The name of the requested meta information. * @return The meta information or NULL if the meta information is not present. */ -CELIX_DFI_EXPORT const char * dynType_getMetaInfo(dyn_type *type, const char *name); +CELIX_DFI_EXPORT const char* dynType_getMetaInfo(const dyn_type* type, const char* name); /** - * Returns a list of meta entries. + * Returns the list of meta entries. * Note that dyn type is owner of the entries. Traversing the entries is not thread safe. * @param type The dyn type. - * @param entries The output arguments for the meta entries. - * @return 0 on success. + * @note It always returns valid list. */ -CELIX_DFI_EXPORT int dynType_metaEntries(dyn_type *type, struct meta_properties_head **entries); +CELIX_DFI_EXPORT const struct meta_properties_head* dynType_metaEntries(const dyn_type* type); /** * Returns the name of the dyn type. Name can be NULL. */ -CELIX_DFI_EXPORT const char * dynType_getName(dyn_type *type); +CELIX_DFI_EXPORT const char* dynType_getName(const dyn_type* type); /** @@ -257,30 +261,27 @@ CELIX_DFI_EXPORT const char * dynType_getName(dyn_type *type); * @param name The field name. * @return The field index or -1 if no field with the name was found. */ -CELIX_DFI_EXPORT int dynType_complex_indexForName(dyn_type *type, const char *name); +CELIX_DFI_EXPORT int dynType_complex_indexForName(const dyn_type* type, const char* name); /** * Returns the field dyn type for a given complex type and a field index. * * @param type The dyn type. Must be a complex type. * @param index The field index for which field the dyn type should be returned. The field index must be correct. - * @param subType The field dyn type as output. - * exists. - * @return 0 if successful. */ -CELIX_DFI_EXPORT int dynType_complex_dynTypeAt(dyn_type *type, int index, dyn_type **subType); +CELIX_DFI_EXPORT const dyn_type* dynType_complex_dynTypeAt(const dyn_type* type, int index); -CELIX_DFI_EXPORT int dynType_complex_setValueAt(dyn_type *type, int index, void *inst, void *in); -CELIX_DFI_EXPORT int dynType_complex_valLocAt(dyn_type *type, int index, void *inst, void **valLoc); -CELIX_DFI_EXPORT int dynType_complex_entries(dyn_type *type, struct complex_type_entries_head **entries); -CELIX_DFI_EXPORT size_t dynType_complex_nrOfEntries(dyn_type *type); +CELIX_DFI_EXPORT int dynType_complex_setValueAt(const dyn_type* type, int index, void* inst, const void* in); +CELIX_DFI_EXPORT void* dynType_complex_valLocAt(const dyn_type* type, int index, void* inst); +CELIX_DFI_EXPORT const struct complex_type_entries_head* dynType_complex_entries(const dyn_type* type); +CELIX_DFI_EXPORT size_t dynType_complex_nrOfEntries(const dyn_type* type); //sequence /** * Initialize a sequence struct with a cap & len of 0 and the buf to NULL. */ -CELIX_DFI_EXPORT void dynType_sequence_init(dyn_type *type, void *inst); +CELIX_DFI_EXPORT void dynType_sequence_init(const dyn_type* type, void* inst); /** * Allocates memory for a sequence with capacity cap. @@ -290,7 +291,7 @@ CELIX_DFI_EXPORT void dynType_sequence_init(dyn_type *type, void *inst); * In case of an error, an error message is added to celix_err. * */ -CELIX_DFI_EXPORT int dynType_sequence_alloc(dyn_type *type, void *inst, uint32_t cap); +CELIX_DFI_EXPORT int dynType_sequence_alloc(const dyn_type* type, void* inst, uint32_t cap); /** * Reserve a sequence capacity of cap @@ -301,7 +302,7 @@ CELIX_DFI_EXPORT int dynType_sequence_alloc(dyn_type *type, void *inst, uint32_t * In case of an error, an error message is added to celix_err. * */ -CELIX_DFI_EXPORT int dynType_sequence_reserve(dyn_type *type, void *inst, uint32_t cap); +CELIX_DFI_EXPORT int dynType_sequence_reserve(const dyn_type* type, void* inst, uint32_t cap); /** * @brief Gets the value location for a specific sequence index from a sequence instance. @@ -315,7 +316,7 @@ CELIX_DFI_EXPORT int dynType_sequence_reserve(dyn_type *type, void *inst, uint32 * @return 0 if successful. * @retval 1 if the index is out of bounds. */ -CELIX_DFI_EXPORT int dynType_sequence_locForIndex(dyn_type *type, void *seqLoc, int index, void **valLoc); +CELIX_DFI_EXPORT int dynType_sequence_locForIndex(const dyn_type* type, const void* seqLoc, uint32_t index, void** valLoc); /** * @brief Increase the length of the sequence by one and return the value location for the last element. @@ -328,29 +329,26 @@ CELIX_DFI_EXPORT int dynType_sequence_locForIndex(dyn_type *type, void *seqLoc, * @return 0 if successful. * @retval 1 if the sequence length is already at the max. */ -CELIX_DFI_EXPORT int dynType_sequence_increaseLengthAndReturnLastLoc(dyn_type *type, void *seqLoc, void **valLoc); +CELIX_DFI_EXPORT int dynType_sequence_increaseLengthAndReturnLastLoc(const dyn_type* type, void* seqLoc, void** valLoc); /** * @brief Get the item type of a sequence. * @param[in] type The dyn type. Must be a sequence type. * @return The item type of the sequence. */ -CELIX_DFI_EXPORT dyn_type * dynType_sequence_itemType(dyn_type *type); +CELIX_DFI_EXPORT const dyn_type* dynType_sequence_itemType(const dyn_type* type); /** * @brief Get the length of a sequence. * @param[in] seqLoc The sequence instance. * @return The length of the sequence. */ -CELIX_DFI_EXPORT uint32_t dynType_sequence_length(void *seqLoc); +CELIX_DFI_EXPORT uint32_t dynType_sequence_length(const void* seqLoc); /** * @brief Gets the typedType of a typedPointer type. - * @param[in] type The dyn type. Must be a typedPointer type. - * @param[out] typedType The typedType of the typedPointer type. - * @return 0 */ -CELIX_DFI_EXPORT int dynType_typedPointer_getTypedType(dyn_type *type, dyn_type **typedType); +CELIX_DFI_EXPORT const dyn_type* dynType_typedPointer_getTypedType(const dyn_type* type); /** * @brief Allocates and initializes a string type. @@ -363,7 +361,7 @@ CELIX_DFI_EXPORT int dynType_typedPointer_getTypedType(dyn_type *type, dyn_type * @return 0 if successful. * @retval 1 if Cannot allocate memory. */ -CELIX_DFI_EXPORT int dynType_text_allocAndInit(dyn_type *type, void *textLoc, const char *value); +CELIX_DFI_EXPORT int dynType_text_allocAndInit(const dyn_type* type, void* textLoc, const char* value); /** * @brief Sets the value of a simple type. @@ -371,7 +369,17 @@ CELIX_DFI_EXPORT int dynType_text_allocAndInit(dyn_type *type, void *textLoc, co * @param[out] inst The instance of the simple type. * @param[in] in The value to set. */ -CELIX_DFI_EXPORT void dynType_simple_setValue(dyn_type *type, void *inst, void *in); +CELIX_DFI_EXPORT void dynType_simple_setValue(const dyn_type* type, void* inst, const void* in); + +/** + * @brief Test whether the given type is trivial. + * A trivial type does NOT involve any pointer type, and thus can always be safely shallow-copied. + * Any non-pointer simple type is trivial, while typed/untyped pointer is non-trivial. + * A sequence is non-trivial since it contains a pointer to the buffer. + * Text is nontrivial. + * A complex is trivial if each field is trivial. + */ +CELIX_DFI_EXPORT bool dynType_isTrivial(const dyn_type* type); #ifdef __cplusplus } diff --git a/libs/dfi/include/json_rpc.h b/libs/dfi/include/json_rpc.h index 931a37dfe..77e8c434e 100644 --- a/libs/dfi/include/json_rpc.h +++ b/libs/dfi/include/json_rpc.h @@ -21,7 +21,6 @@ #define __JSON_RPC_H_ #include -#include "dfi_log_util.h" #include "dyn_type.h" #include "dyn_function.h" #include "dyn_interface.h" @@ -31,9 +30,6 @@ extern "C" { #endif -//logging -DFI_SETUP_LOG_HEADER(jsonRpc); - /** * @brief Call a remote service using JSON-RPC. * @@ -48,7 +44,7 @@ DFI_SETUP_LOG_HEADER(jsonRpc); * @return 0 if successful, otherwise 1. * */ -CELIX_DFI_EXPORT int jsonRpc_call(dyn_interface_type *intf, void *service, const char *request, char **out); +CELIX_DFI_EXPORT int jsonRpc_call(const dyn_interface_type* intf, void* service, const char* request, char** out); /** * @brief Prepare a JSON-RPC request for a given function. @@ -64,7 +60,7 @@ CELIX_DFI_EXPORT int jsonRpc_call(dyn_interface_type *intf, void *service, const * @return 0 if successful, otherwise 1. * */ -CELIX_DFI_EXPORT int jsonRpc_prepareInvokeRequest(dyn_function_type *func, const char *id, void *args[], char **out); +CELIX_DFI_EXPORT int jsonRpc_prepareInvokeRequest(const dyn_function_type* func, const char* id, void* args[], char** out); /** * @brief Handle a JSON-RPC reply for a given function. @@ -78,7 +74,7 @@ CELIX_DFI_EXPORT int jsonRpc_prepareInvokeRequest(dyn_function_type *func, const * @return 0 if successful, otherwise 1. * */ -CELIX_DFI_EXPORT int jsonRpc_handleReply(dyn_function_type *func, const char *reply, void *args[], int *rsErrno); +CELIX_DFI_EXPORT int jsonRpc_handleReply(const dyn_function_type* func, const char* reply, void* args[], int* rsErrno); #ifdef __cplusplus } diff --git a/libs/dfi/include/json_serializer.h b/libs/dfi/include/json_serializer.h index 6ef61707d..1c4cb44c7 100644 --- a/libs/dfi/include/json_serializer.h +++ b/libs/dfi/include/json_serializer.h @@ -21,7 +21,6 @@ #define __JSON_SERIALIZER_H_ #include -#include "dfi_log_util.h" #include "dyn_type.h" #include "dyn_function.h" #include "dyn_interface.h" @@ -31,23 +30,22 @@ extern "C" { #endif -//logging -DFI_SETUP_LOG_HEADER(jsonSerializer); - /** - * @brief Deserialize a JSON string to a given type. + * @brief Deserialize a JSON string buffer to a given type. + * @note The string buffer doesn't need to be null-terminated. * * Caller is the owner of the out parameter and should release it using dynType_free. * * In case of an error, an error message is added to celix_err. * * @param[in] type The type to deserialize to. - * @param[in] input The JSON string to deserialize. - * @param[out] out The deserialized result. + * @param[in] input The JSON string buffer to deserialize. + * @param[in] length The length of the given JSON string buffer. + * @param[out] result The deserialized result. * @return 0 if successful, otherwise 1. * */ -CELIX_DFI_EXPORT int jsonSerializer_deserialize(dyn_type *type, const char *input, size_t length, void **result); +CELIX_DFI_EXPORT int jsonSerializer_deserialize(const dyn_type* type, const char* input, size_t length, void** result); /** * @brief Deserialize a JSON object to a given type. @@ -62,7 +60,7 @@ CELIX_DFI_EXPORT int jsonSerializer_deserialize(dyn_type *type, const char *inpu * @return 0 if successful, otherwise 1. * */ -CELIX_DFI_EXPORT int jsonSerializer_deserializeJson(dyn_type *type, json_t *input, void **result); +CELIX_DFI_EXPORT int jsonSerializer_deserializeJson(const dyn_type* type, json_t* input, void** result); /** * @brief Serialize a given type to a JSON string. @@ -77,7 +75,7 @@ CELIX_DFI_EXPORT int jsonSerializer_deserializeJson(dyn_type *type, json_t *inpu * @return 0 if successful, otherwise 1. * */ -CELIX_DFI_EXPORT int jsonSerializer_serialize(dyn_type *type, const void* input, char **output); +CELIX_DFI_EXPORT int jsonSerializer_serialize(const dyn_type* type, const void* input, char** output); /** * @brief Serialize a given type to a JSON object. @@ -92,7 +90,7 @@ CELIX_DFI_EXPORT int jsonSerializer_serialize(dyn_type *type, const void* input, * @return 0 if successful, otherwise 1. * */ -CELIX_DFI_EXPORT int jsonSerializer_serializeJson(dyn_type *type, const void* input, json_t **out); +CELIX_DFI_EXPORT int jsonSerializer_serializeJson(const dyn_type* type, const void* input, json_t** out); #ifdef __cplusplus } diff --git a/libs/dfi/src/dyn_common.c b/libs/dfi/src/dyn_common.c index adc33c8dc..4a27c1a55 100644 --- a/libs/dfi/src/dyn_common.c +++ b/libs/dfi/src/dyn_common.c @@ -18,124 +18,90 @@ */ #include "dyn_common.h" +#include "celix_err.h" +#include "celix_stdio_cleanup.h" +#include "celix_stdlib_cleanup.h" #include #include #include #include -#if CELIX_UTILS_NO_MEMSTREAM_AVAILABLE -#include "open_memstream.h" -#include "fmemopen.h" -#endif - static const int OK = 0; static const int ERROR = 1; -DFI_SETUP_LOG(dynCommon) - -static bool dynCommon_charIn(int c, const char *acceptedChars); -int dynCommon_parseName(FILE *stream, char **result) { +int dynCommon_parseName(FILE* stream, char** result) { return dynCommon_parseNameAlsoAccept(stream, NULL, result); } -int dynCommon_parseNameAlsoAccept(FILE *stream, const char *acceptedChars, char **result) { - int status = OK; - - char *buf = NULL; +int dynCommon_parseNameAlsoAccept(FILE* stream, const char* acceptedChars, char** result) { + celix_autofree char* buf = NULL; size_t size = 0; - int strLen = 0; - FILE *name = open_memstream(&buf, &size); - - if (name != NULL) { - int c = getc(stream); - while (isalnum(c) || c == '_' || dynCommon_charIn(c, acceptedChars)) { - fputc(c, name); - c = getc(stream); - strLen += 1; - } - fflush(name); - fclose(name); - ungetc(c, stream); - } else { - status = ERROR; - LOG_ERROR("Error creating mem stream for name. %s", strerror(errno)); + celix_autoptr(FILE) name = open_memstream(&buf, &size); + if (name == NULL) { + celix_err_pushf("Error creating mem stream for name. %s", strerror(errno)); + return ERROR; } - if (status == OK) { - if (strLen == 0) { - status = ERROR; - LOG_ERROR("Parsed empty name"); + int c = getc(stream); + while (c != EOF && (isalnum(c) || c == '_' || (acceptedChars != NULL && strchr(acceptedChars, c) != NULL))) { + if(fputc(c, name) == EOF) { + celix_err_push("Error writing to mem stream for name."); + return ERROR; } + c = getc(stream); + } + if (c != EOF) { + ungetc(c, stream); + } + if(fclose(celix_steal_ptr(name)) != 0) { + celix_err_pushf("Error closing mem stream for name. %s", strerror(errno)); + return ERROR; } - if (status == OK) { - *result = buf; - } else if (buf != NULL) { - free(buf); + if (size == 0) { + celix_err_push("Parsed empty name"); + return ERROR; } - return status; + *result = celix_steal_ptr(buf); + + return OK; } -int dynCommon_parseNameValue(FILE *stream, char **outName, char **outValue) { +static int dynCommon_parseNameValue(FILE* stream, char** outName, char** outValue) { int status; - char *name = NULL; - char *value = NULL; - - status = dynCommon_parseName(stream, &name); - if (status == OK) { - status = dynCommon_eatChar(stream, '='); + celix_autofree char* name = NULL; + celix_autofree char* value = NULL; + if ((status = dynCommon_parseName(stream, &name)) != OK) { + return status; } - if (status == OK) { - const char *valueAcceptedChars = ".<>{}[]?;:~!@#$%^&*()_+-=,./\\'\""; - - status = dynCommon_parseNameAlsoAccept(stream, valueAcceptedChars, &value); //NOTE use different more lenient function e.g. only stop at '\n' ? + if ((status = dynCommon_eatChar(stream, '=')) != OK) { + return status; } - - if (status == OK) { - *outName = name; - *outValue = value; - } else { - if (name != NULL) { - free(name); - } - if (value != NULL) { - free(value); - } + const char *valueAcceptedChars = ".<>{}[]?;:~!@#$%^&*()_+-=,./\\'\""; + //NOTE use different more lenient function e.g. only stop at '\n' ? + if ((status = dynCommon_parseNameAlsoAccept(stream, valueAcceptedChars, &value)) != OK) { + return status; } - return status; + *outName = celix_steal_ptr(name); + *outValue = celix_steal_ptr(value); + return OK; } -int dynCommon_eatChar(FILE *stream, int expected) { +int dynCommon_eatChar(FILE* stream, int expected) { int status = OK; - long loc = ftell(stream); int c = fgetc(stream); if (c != expected) { status = ERROR; - LOG_ERROR("Error parsing, expected token '%c' got '%c' at position %li", expected, c, loc); - } - return status; -} - -static bool dynCommon_charIn(int c, const char *acceptedChars) { - bool status = false; - if (acceptedChars != NULL) { - int i; - for (i = 0; acceptedChars[i] != '\0'; i += 1) { - if (c == acceptedChars[i]) { - status = true; - break; - } - } + celix_err_pushf("Error parsing, expected token '%c' got '%c' at position %li", expected, c, ftell(stream)); } - return status; } -void dynCommon_clearNamValHead(struct namvals_head *head) { - struct namval_entry *entry = TAILQ_FIRST(head); +void dynCommon_clearNamValHead(struct namvals_head* head) { + struct namval_entry* entry = TAILQ_FIRST(head); while (entry != NULL) { struct namval_entry *tmp = entry; @@ -149,3 +115,59 @@ void dynCommon_clearNamValHead(struct namvals_head *head) { free(tmp); } } + +int dynCommon_parseNameValueSection(FILE* stream, struct namvals_head* head) { + int status = OK; + + int peek = fgetc(stream); + while (peek != ':' && peek != EOF) { + ungetc(peek, stream); + + celix_autofree char* name = NULL; + celix_autofree char* value = NULL; + if ((status = dynCommon_parseNameValue(stream, &name, &value)) != OK) { + return status; + } + + if ((status = dynCommon_eatChar(stream, '\n')) != OK) { + return status; + } + + struct namval_entry* entry = NULL; + entry = calloc(1, sizeof(*entry)); + if (entry == NULL) { + celix_err_pushf("Error allocating memory for namval entry"); + return ERROR; + } + + entry->name = celix_steal_ptr(name); + entry->value = celix_steal_ptr(value); + TAILQ_INSERT_TAIL(head, entry, entries); + + peek = fgetc(stream); + } + if (peek != EOF) { + ungetc(peek, stream); + } + + return OK; +} + +int dynCommon_getEntryForHead(const struct namvals_head* head, const char* name, const char** out) { + int status = OK; + char* value = NULL; + struct namval_entry* entry = NULL; + TAILQ_FOREACH(entry, head, entries) { + if (strcmp(name, entry->name) == 0) { + value = entry->value; + break; + } + } + if (value != NULL) { + *out = value; + } else { + status = ERROR; + celix_err_pushf("Cannot find '%s' in list", name); + } + return status; +} diff --git a/libs/dfi/include/dyn_common.h b/libs/dfi/src/dyn_common.h similarity index 71% rename from libs/dfi/include/dyn_common.h rename to libs/dfi/src/dyn_common.h index 1c44280bf..66dc1c8f2 100644 --- a/libs/dfi/include/dyn_common.h +++ b/libs/dfi/src/dyn_common.h @@ -25,28 +25,22 @@ #include #include -#include "dfi_log_util.h" -#include "celix_dfi_export.h" - #ifdef __cplusplus extern "C" { #endif -//logging -DFI_SETUP_LOG_HEADER(dynCommon) ; - TAILQ_HEAD(namvals_head, namval_entry); struct namval_entry { - char *name; - char *value; + char* name; + char* value; TAILQ_ENTRY(namval_entry) entries; }; /** * @brief Parse the name of dynamic type from the given stream. The name is only allowed to contain [a-zA-Z0-9_]. * - * The caller is the owner of the dynamic type name and use `free` deallocate the memory. + * The caller is the owner of the dynamic type name and use `free` to deallocate the memory. * * In case of an error, an error message is added to celix_err. * @@ -55,7 +49,7 @@ struct namval_entry { * @return 0 if successful, otherwise 1. * @alsoseee dynCommon_parseNameAlsoAccept */ -CELIX_DFI_EXPORT int dynCommon_parseName(FILE *stream, char **result); +int dynCommon_parseName(FILE* stream, char** result); /** * @brief Parse the name of dynamic type from the given stream. @@ -70,21 +64,24 @@ CELIX_DFI_EXPORT int dynCommon_parseName(FILE *stream, char **result); * @return 0 if successful, otherwise 1. * @alsoseee dynCommon_parseName */ -CELIX_DFI_EXPORT int dynCommon_parseNameAlsoAccept(FILE *stream, const char *acceptedChars, char **result); +int dynCommon_parseNameAlsoAccept(FILE* stream, const char* acceptedChars, char** result); /** - * @brief Parse the name and value of a name-value pair from the given stream. The name is only allowed to contain [a-zA-Z0-9_]. + * @brief Parses a section of name-value pairs from the given stream. The name is only allowed to contain [a-zA-Z0-9_]. * - * The caller is the owner of the name and value and use `free` deallocate the memory. + * This function reads from the provided stream until it encounters a new section or EOF. + * Each name-value pair is added to the provided namvals_head structure. + * + * The caller is responsible for managing the memory of the namvals_head structure. + * Use `dynCommon_clearNamValHead` to clear the name-value pairs when they are no longer needed. * * In case of an error, an error message is added to celix_err. * * @param[in] stream The input stream. - * @param[out] name The name of the name-value pair. - * @param[out] value The value of the name-value pair. + * @param[out] head The namvals_head structure where the parsed name-value pairs will be stored. * @return 0 if successful, otherwise 1. */ -CELIX_DFI_EXPORT int dynCommon_parseNameValue(FILE *stream, char **name, char **value); +int dynCommon_parseNameValueSection(FILE* stream, struct namvals_head* head); /** * @brief Eat the given character from the given stream. @@ -95,14 +92,16 @@ CELIX_DFI_EXPORT int dynCommon_parseNameValue(FILE *stream, char **name, char ** * @param[in] c The character to be eaten. * @return 0 if successful, otherwise 1. */ -CELIX_DFI_EXPORT int dynCommon_eatChar(FILE *stream, int c); +int dynCommon_eatChar(FILE* stream, int c); /** * @brief Clear the given name-value pairs. * * @param[in] head The name-value pairs to be cleared. */ -CELIX_DFI_EXPORT void dynCommon_clearNamValHead(struct namvals_head *head); +void dynCommon_clearNamValHead(struct namvals_head* head); + +int dynCommon_getEntryForHead(const struct namvals_head* head, const char* name, const char** out); #ifdef __cplusplus } diff --git a/libs/dfi/src/dyn_descriptor.c b/libs/dfi/src/dyn_descriptor.c new file mode 100644 index 000000000..be5ad8fa1 --- /dev/null +++ b/libs/dfi/src/dyn_descriptor.c @@ -0,0 +1,178 @@ +/* + * 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. + */ + +#include "dyn_descriptor.h" +#include "celix_err.h" +#include "celix_stdlib_cleanup.h" + +#include + +static const int OK = 0; +static const int ERROR = 1; + +static int celix_dynDescriptor_parseTypes(celix_descriptor_t* descriptor, FILE *stream) { + int status; + + //expected input (Name)=\n + int peek = fgetc(stream); + while (peek != ':' && peek != EOF) { + ungetc(peek, stream); + + celix_autofree char* name = NULL; + if ((status = dynCommon_parseName(stream, &name)) != OK) { + return status; + } + + if ((status = dynCommon_eatChar(stream, '=')) != OK) { + return status; + } + + celix_autoptr(dyn_type) type = NULL; + if ((status = dynType_parseOfName(stream, celix_steal_ptr(name), &descriptor->types, &type)) != OK) { + return status; + } + if ((status = dynCommon_eatChar(stream, '\n')) != OK) { + return status; + } + + struct type_entry *entry = NULL; + entry = calloc(1, sizeof(*entry)); + if (entry == NULL) { + celix_err_pushf("Error allocating memory for type entry"); + return ERROR; + } + entry->type = celix_steal_ptr(type); + TAILQ_INSERT_TAIL(&descriptor->types, entry, entries); + + peek = fgetc(stream); + } + if (peek != EOF) { + ungetc(peek, stream); + } + + return OK; +} + +static int celix_dynDescriptor_parseSection(celix_descriptor_t* descriptor, FILE *stream, + int (*parseSection)(celix_descriptor_t* descriptor, const char* secName, FILE *stream)) { + int status = OK; + celix_autofree char *sectionName = NULL; + + if ((status = dynCommon_eatChar(stream, ':')) != OK) { + return status; + } + + if ((status = dynCommon_parseName(stream, §ionName)) != OK) { + return status; + } + + if ((status = dynCommon_eatChar(stream, '\n')) != OK) { + return status; + } + + if (strcmp("header", sectionName) == 0) { + status = dynCommon_parseNameValueSection(stream, &descriptor->header); + } else if (strcmp("annotations", sectionName) == 0) { + status = dynCommon_parseNameValueSection(stream, &descriptor->annotations); + } else if (strcmp("types", sectionName) == 0) { + status = celix_dynDescriptor_parseTypes(descriptor, stream); + } else { + status = parseSection(descriptor, sectionName, stream); + } + + return status; +} + +static int celix_dynDescriptor_checkInterface(celix_descriptor_t* descriptor) { + //check header section + bool foundType = false; + bool foundVersion = false; + bool foundName = false; + struct namval_entry *entry = NULL; + TAILQ_FOREACH(entry, &descriptor->header, entries) { + if (strcmp(entry->name, "type") == 0) { + foundType = true; + } else if (strcmp(entry->name, "version") == 0) { + foundVersion = true; + } else if (strcmp(entry->name, "name") == 0) { + foundName = true; + } + } + + if (!foundType || !foundVersion || !foundName) { + celix_err_pushf("Parse Error. There must be a header section with a type, version and name entry"); + return ERROR; + } + + return OK; +} + +int celix_dynDescriptor_parse(celix_descriptor_t* descriptor, FILE* stream, + int (*parseSection)(celix_descriptor_t* descriptor, const char* secName, FILE *stream)) { + int status = OK; + + char peek = (char)fgetc(stream); + while (peek == ':') { + ungetc(peek, stream); + if ((status = celix_dynDescriptor_parseSection(descriptor, stream, parseSection)) != OK) { + return status; + } + peek = (char)fgetc(stream); + } + + if (peek != EOF) { + celix_err_pushf("Descriptor does not start with ':'"); + return ERROR; + } + + if ((status = celix_dynDescriptor_checkInterface(descriptor)) != OK) { + return status; + } + + // We are sure that version field is present in the header + const char* version; + status = dynCommon_getEntryForHead(&descriptor->header, "version", &version); + assert(status == OK); + assert(version != NULL); + descriptor->version = celix_version_createVersionFromString(version); + if (descriptor->version == NULL) { + celix_err_pushf("Invalid version (%s) in parsed descriptor\n",version); + return ERROR; + } + return OK; +} + +void celix_dynDescriptor_destroy(celix_descriptor_t* descriptor) { + dynCommon_clearNamValHead(&descriptor->header); + dynCommon_clearNamValHead(&descriptor->annotations); + + struct type_entry *tInfo = TAILQ_FIRST(&descriptor->types); + while (tInfo != NULL) { + struct type_entry *tmp = tInfo; + tInfo = TAILQ_NEXT(tInfo, entries); + dynType_destroy(tmp->type); + free(tmp); + } + + if(descriptor->version!=NULL){ + celix_version_destroy(descriptor->version); + } + + free(descriptor); +} diff --git a/libs/dfi/src/dyn_descriptor.h b/libs/dfi/src/dyn_descriptor.h new file mode 100644 index 000000000..e7d085f76 --- /dev/null +++ b/libs/dfi/src/dyn_descriptor.h @@ -0,0 +1,51 @@ +/* + * 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. + */ + +#ifndef CELIX_DYN_DESCRIPTOR_H +#define CELIX_DYN_DESCRIPTOR_H +#ifdef __cplusplus +extern "C" { +#endif + +#include "dyn_common.h" +#include "dyn_type.h" +#include "celix_version.h" + +#include + +#define CELIX_DESCRIPTOR_FIELDS \ + /* public */ \ + struct namvals_head header; \ + struct namvals_head annotations; \ + struct types_head types; \ + celix_version_t* version; \ + +typedef struct celix_descriptor { + CELIX_DESCRIPTOR_FIELDS +}celix_descriptor_t; + +int celix_dynDescriptor_parse(celix_descriptor_t* descriptor, FILE* stream, + int (*parseSection)(celix_descriptor_t* descriptor, const char* secName, FILE *stream)); + +void celix_dynDescriptor_destroy(celix_descriptor_t* descriptor); + +#ifdef __cplusplus +} +#endif +#endif //CELIX_DYN_DESCRIPTOR_H diff --git a/libs/dfi/src/dyn_function.c b/libs/dfi/src/dyn_function.c index 1b6666cc8..3bf27d826 100644 --- a/libs/dfi/src/dyn_function.c +++ b/libs/dfi/src/dyn_function.c @@ -19,6 +19,9 @@ #include "dyn_function.h" #include "dyn_function_common.h" +#include "celix_cleanup.h" +#include "celix_err.h" +#include "celix_stdlib_cleanup.h" #include #include @@ -30,141 +33,134 @@ static const int MEM_ERROR = 1; static const int PARSE_ERROR = 2; static const int ERROR = 2; -DFI_SETUP_LOG(dynFunction) +static int dynFunction_initCif(dyn_function_type* dynFunc); +static int dynFunction_parseDescriptor(dyn_function_type* dynFunc, FILE* descriptor); +static void dynFunction_ffiBind(ffi_cif* cif, void* ret, void* args[], void* userData); -static int dynFunction_initCif(dyn_function_type *dynFunc); -static int dynFunction_parseDescriptor(dyn_function_type *dynFunc, FILE *descriptor); -static void dynFunction_ffiBind(ffi_cif *cif, void *ret, void *args[], void *userData); - -ffi_type * dynType_ffiType(dyn_type *type); - -int dynFunction_parse(FILE *descriptor, struct types_head *refTypes, dyn_function_type **out) { +int dynFunction_parse(FILE* descriptor, struct types_head* refTypes, dyn_function_type** out) { int status = OK; - dyn_function_type *dynFunc = NULL; - - dynFunc = calloc(1, sizeof(*dynFunc)); + celix_autoptr(dyn_function_type) dynFunc = calloc(1, sizeof(*dynFunc)); + if (dynFunc == NULL) { + celix_err_pushf("Error allocating memory for dyn function"); + return MEM_ERROR; + } - if (dynFunc != NULL) { - TAILQ_INIT(&dynFunc->arguments); - dynFunc->refTypes = refTypes; - status = dynFunction_parseDescriptor(dynFunc, descriptor); - if (status == 0) { - int rc = dynFunction_initCif(dynFunc); - if (rc != 0) { - LOG_ERROR("Error initializing cif"); - status = ERROR; - } - } - } else { - LOG_ERROR("Error allocating memory for dyn function\n"); - status = MEM_ERROR; + TAILQ_INIT(&dynFunc->arguments); + dynFunc->refTypes = refTypes; + status = dynFunction_parseDescriptor(dynFunc, descriptor); + if (status != OK) { + celix_err_pushf("Error parsing descriptor"); + return status; + } + status = dynFunction_initCif(dynFunc); + if (status != OK) { + return status; } - if (status == OK) { - dyn_function_argument_type *arg = NULL; - TAILQ_FOREACH(arg, &dynFunc->arguments, entries) { - const char *meta = dynType_getMetaInfo(arg->type, "am"); - if (meta == NULL) { - arg->argumentMeta = DYN_FUNCTION_ARGUMENT_META__STD; - } else if (strcmp(meta, "handle") == 0) { - arg->argumentMeta = DYN_FUNCTION_ARGUMENT_META__HANDLE; - } else if (strcmp(meta, "pre") == 0) { - arg->argumentMeta = DYN_FUNCTION_ARGUMENT_META__PRE_ALLOCATED_OUTPUT; - } else if (strcmp(meta, "out") == 0) { - arg->argumentMeta = DYN_FUNCTION_ARGUMENT_META__OUTPUT; - } else { - arg->argumentMeta = DYN_FUNCTION_ARGUMENT_META__STD; + dyn_function_argument_type* arg = NULL; + TAILQ_FOREACH(arg, &dynFunc->arguments, entries) { + const dyn_type* real = dynType_realType(arg->type); + const char* meta = dynType_getMetaInfo(arg->type, "am"); + if (meta == NULL) { + arg->argumentMeta = DYN_FUNCTION_ARGUMENT_META__STD; + } else if (strcmp(meta, "handle") == 0) { + if (dynType_descriptorType(real) != 'P') { + celix_err_pushf("Error 'handle' is only allowed for untyped pointer not '%c'", dynType_descriptorType(real)); + return PARSE_ERROR; + } + arg->argumentMeta = DYN_FUNCTION_ARGUMENT_META__HANDLE; + } else if (strcmp(meta, "pre") == 0) { + if (dynType_type(real) != DYN_TYPE_TYPED_POINTER) { + celix_err_pushf("Error 'pre' is only allowed for typed pointer not '%c'", dynType_descriptorType(real)); + return PARSE_ERROR; + } + const dyn_type* sub = dynType_typedPointer_getTypedType(real); + if (!dynType_isTrivial(sub)) { + celix_err_pushf("Error 'pre' is only allowed for pointer to trivial types not non-trivial '%c'", dynType_descriptorType(sub)); + return PARSE_ERROR; } + arg->argumentMeta = DYN_FUNCTION_ARGUMENT_META__PRE_ALLOCATED_OUTPUT; + } else if (strcmp(meta, "out") == 0) { + if (dynType_type(real) != DYN_TYPE_TYPED_POINTER) { + celix_err_pushf("Error 'out' is only allowed for typed pointer not '%c'", dynType_descriptorType(real)); + return PARSE_ERROR; + } + const dyn_type* sub = dynType_typedPointer_getTypedType(real); + int subType = dynType_type(sub); + if (subType != DYN_TYPE_TEXT && subType != DYN_TYPE_TYPED_POINTER) { + celix_err_pushf("Error 'out' is only allowed for pointer to text or typed pointer not to '%c'", + dynType_descriptorType(sub)); + return PARSE_ERROR; + } + arg->argumentMeta = DYN_FUNCTION_ARGUMENT_META__OUTPUT; + } else { + arg->argumentMeta = DYN_FUNCTION_ARGUMENT_META__STD; } } - - if (status == OK) { - *out = dynFunc; - } else { - LOG_ERROR("Failed to Create dyn function"); - if (dynFunc != NULL) { - dynFunction_destroy(dynFunc); - } - } - - return status; + *out = celix_steal_ptr(dynFunc); + return OK; } -int dynFunction_parseWithStr(const char *descriptor, struct types_head *refTypes, dyn_function_type **out) { +int dynFunction_parseWithStr(const char* descriptor, struct types_head* refTypes, dyn_function_type** out) { int status = OK; - FILE *stream = fmemopen((char *)descriptor, strlen(descriptor) + 1, "r"); - if (stream != NULL) { - status = dynFunction_parse(stream, refTypes, out); - fclose(stream); - } else { - status = MEM_ERROR; - LOG_ERROR("Error creating mem stream for descriptor string. %s", strerror(errno)); + FILE* stream = fmemopen((char* )descriptor, strlen(descriptor), "r"); + if (stream == NULL) { + celix_err_pushf("Error creating mem stream for descriptor string. %s", strerror(errno)); + return MEM_ERROR; } + status = dynFunction_parse(stream, refTypes, out); + fclose(stream); return status; } -static int dynFunction_parseDescriptor(dyn_function_type *dynFunc, FILE *descriptor) { +static int dynFunction_parseDescriptor(dyn_function_type* dynFunc, FILE* descriptor) { int status = OK; - char *name = NULL; - status = dynCommon_parseName(descriptor, &name); - - if (status == OK) { - dynFunc->name = name; + if ((status = dynCommon_parseName(descriptor, &dynFunc->name)) != OK) { + return status; } - if (status == OK) { - int c = fgetc(descriptor); - if ( c != '(') { - status = PARSE_ERROR; - LOG_ERROR("Expected '(' token got '%c'", c); - } + status = dynCommon_eatChar(descriptor, '('); + if (status != OK) { + return PARSE_ERROR; } int nextChar = fgetc(descriptor); int index = 0; - dyn_type *type = NULL; - char argName[32]; - while (nextChar != ')' && status == 0) { + while (nextChar != ')' && nextChar != EOF) { ungetc(nextChar, descriptor); - type = NULL; - - dyn_function_argument_type *arg = NULL; - - status = dynType_parse(descriptor, NULL, dynFunc->refTypes, &type); - if (status == OK) { - arg = calloc(1, sizeof(*arg)); - if (arg != NULL) { - arg->index = index; - arg->type = type; - snprintf(argName, 32, "arg%04i", index); - arg->name = strdup(argName); - - index += 1; - } else { - LOG_ERROR("Error allocating memory"); - status = MEM_ERROR; - } + + celix_autoptr(dyn_type) type = NULL; + dyn_function_argument_type* arg = NULL; + + if ((status = dynType_parse(descriptor, NULL, dynFunc->refTypes, &type)) != OK) { + return status; } - if (status == OK) { - TAILQ_INSERT_TAIL(&dynFunc->arguments, arg, entries); + arg = calloc(1, sizeof(*arg)); + if (arg == NULL) { + celix_err_pushf("Error allocating arg"); + return MEM_ERROR; } + arg->index = index++; + arg->type = celix_steal_ptr(type); + + TAILQ_INSERT_TAIL(&dynFunc->arguments, arg, entries); nextChar = fgetc(descriptor); } - - if (status == 0) { - status = dynType_parse(descriptor, NULL, dynFunc->refTypes, &dynFunc->funcReturn); + if (nextChar != ')') { + celix_err_push("Error missing ')'"); + return PARSE_ERROR; } - - return status; + + return dynType_parse(descriptor, NULL, dynFunc->refTypes, &dynFunc->funcReturn); } -enum dyn_function_argument_meta dynFunction_argumentMetaForIndex(dyn_function_type *dynFunc, int argumentNr) { +enum dyn_function_argument_meta dynFunction_argumentMetaForIndex(const dyn_function_type* dynFunc, int argumentNr) { enum dyn_function_argument_meta result = 0; - dyn_function_argument_type *arg = NULL; + dyn_function_argument_type* arg = NULL; int index = 0; TAILQ_FOREACH(arg, &dynFunc->arguments, entries) { if (index == argumentNr) { @@ -176,121 +172,116 @@ enum dyn_function_argument_meta dynFunction_argumentMetaForIndex(dyn_function_ty return result; } +const struct dyn_function_arguments_head* dynFunction_arguments(const dyn_function_type* dynFunc) { + return &dynFunc->arguments; +} -static int dynFunction_initCif(dyn_function_type *dynFunc) { - int status = 0; - +static int dynFunction_initCif(dyn_function_type* dynFunc) { unsigned int nargs = 0; - dyn_function_argument_type *entry = NULL; + dyn_function_argument_type* entry = NULL; TAILQ_FOREACH(entry, &dynFunc->arguments, entries) { nargs +=1; } dynFunc->ffiArguments = calloc(nargs, sizeof(ffi_type*)); + if (dynFunc->ffiArguments == NULL) { + celix_err_push("Error allocating memory for ffi args"); + return MEM_ERROR; + } TAILQ_FOREACH(entry, &dynFunc->arguments, entries) { dynFunc->ffiArguments[entry->index] = dynType_ffiType(entry->type); } - ffi_type **args = dynFunc->ffiArguments; - ffi_type *returnType = dynType_ffiType(dynFunc->funcReturn); + ffi_type** args = dynFunc->ffiArguments; + ffi_type* returnType = dynType_ffiType(dynFunc->funcReturn); int ffiResult = ffi_prep_cif(&dynFunc->cif, FFI_DEFAULT_ABI, nargs, returnType, args); if (ffiResult != FFI_OK) { - status = 1; + celix_err_pushf("Error ffi_prep_cif %d", ffiResult); + return ERROR; } - return status; + return OK; } -void dynFunction_destroy(dyn_function_type *dynFunc) { +void dynFunction_destroy(dyn_function_type* dynFunc) { if (dynFunc != NULL) { - if (dynFunc->funcReturn != NULL) { - dynType_destroy(dynFunc->funcReturn); - } + // release resource in strict reverse order if (dynFunc->ffiClosure != NULL) { ffi_closure_free(dynFunc->ffiClosure); } - if (dynFunc->name != NULL) { - free(dynFunc->name); - } if (dynFunc->ffiArguments != NULL) { free(dynFunc->ffiArguments); } - - dyn_function_argument_type *entry = NULL; - dyn_function_argument_type *tmp = NULL; - entry = TAILQ_FIRST(&dynFunc->arguments); + if (dynFunc->funcReturn != NULL) { + dynType_destroy(dynFunc->funcReturn); + } + dyn_function_argument_type* entry = NULL; + dyn_function_argument_type* tmp = NULL; + entry = TAILQ_FIRST(&dynFunc->arguments); while (entry != NULL) { - if (entry->name != NULL) { - free(entry->name); - } dynType_destroy(entry->type); tmp = entry; entry = TAILQ_NEXT(entry, entries); free(tmp); } - + if (dynFunc->name != NULL) { + free(dynFunc->name); + } free(dynFunc); } } -int dynFunction_call(dyn_function_type *dynFunc, void(*fn)(void), void *returnValue, void **argValues) { - ffi_call(&dynFunc->cif, fn, returnValue, argValues); +int dynFunction_call(const dyn_function_type* dynFunc, void(*fn)(void), void* returnValue, void** argValues) { + ffi_call((ffi_cif*)&dynFunc->cif, fn, returnValue, argValues); return 0; } -static void dynFunction_ffiBind(ffi_cif *cif, void *ret, void *args[], void *userData) { - dyn_function_type *dynFunc = userData; +static void dynFunction_ffiBind(ffi_cif* cif, void* ret, void* args[], void* userData) { + dyn_function_type* dynFunc = userData; dynFunc->bind(dynFunc->userData, args, ret); } -int dynFunction_createClosure(dyn_function_type *dynFunc, void (*bind)(void *, void **, void*), void *userData, void(**out)(void)) { - int status = 0; +CELIX_DEFINE_AUTOPTR_CLEANUP_FUNC(ffi_closure, ffi_closure_free) + +int dynFunction_createClosure(dyn_function_type* dynFunc, void (*bind)(void*, void**, void*), void* userData, void(**out)(void)) { void (*fn)(void); - dynFunc->ffiClosure = ffi_closure_alloc(sizeof(ffi_closure), (void **)&fn); - if (dynFunc->ffiClosure != NULL) { - int rc = ffi_prep_closure_loc(dynFunc->ffiClosure, &dynFunc->cif, dynFunction_ffiBind, dynFunc, fn); - if (rc != FFI_OK) { - status = 1; - } - } else { - status = 2; + celix_autoptr(ffi_closure) ffiClosure = ffi_closure_alloc(sizeof(ffi_closure), (void **)&fn); + if (ffiClosure == NULL) { + return MEM_ERROR; } - - if (status == 0) { - dynFunc->userData = userData; - dynFunc->bind = bind; - dynFunc->fn = fn; - *out =fn; + int rc = ffi_prep_closure_loc(ffiClosure, &dynFunc->cif, dynFunction_ffiBind, dynFunc, fn); + if (rc != FFI_OK) { + return ERROR; } - return status; + dynFunc->ffiClosure = celix_steal_ptr(ffiClosure); + dynFunc->userData = userData; + dynFunc->bind = bind; + dynFunc->fn = fn; + *out =fn; + + return OK; } -int dynFunction_getFnPointer(dyn_function_type *dynFunc, void (**fn)(void)) { - int status = 0; - if (dynFunc != NULL && dynFunc->fn != NULL) { - (*fn) = dynFunc->fn; - } else { - status = 1; +int dynFunction_getFnPointer(const dyn_function_type* dynFunc, void (**fn)(void)) { + if (dynFunc == NULL || dynFunc->fn == NULL) { + return ERROR; } - return status; + (*fn) = dynFunc->fn; + return OK; } -int dynFunction_nrOfArguments(dyn_function_type *dynFunc) { - int count = 0; - dyn_function_argument_type *entry = NULL; - TAILQ_FOREACH(entry, &dynFunc->arguments, entries) { - count += 1; - } - return count; +int dynFunction_nrOfArguments(const dyn_function_type* dynFunc) { + dyn_function_argument_type* last = TAILQ_LAST(&dynFunc->arguments, dyn_function_arguments_head); + return last == NULL ? 0 : (last->index+1); } -dyn_type *dynFunction_argumentTypeForIndex(dyn_function_type *dynFunc, int argumentNr) { - dyn_type *result = NULL; +const dyn_type* dynFunction_argumentTypeForIndex(const dyn_function_type* dynFunc, int argumentNr) { + dyn_type* result = NULL; int index = 0; - dyn_function_argument_type *entry = NULL; + dyn_function_argument_type* entry = NULL; TAILQ_FOREACH(entry, &dynFunc->arguments, entries) { if (index == argumentNr) { result = entry->type; @@ -301,11 +292,15 @@ dyn_type *dynFunction_argumentTypeForIndex(dyn_function_type *dynFunc, int argum return result; } -dyn_type * dynFunction_returnType(dyn_function_type *dynFunction) { +const dyn_type* dynFunction_returnType(const dyn_function_type *dynFunction) { return dynFunction->funcReturn; } -bool dynFunction_hasReturn(dyn_function_type *dynFunction) { - dyn_type *t = dynFunction_returnType(dynFunction); +bool dynFunction_hasReturn(const dyn_function_type* dynFunction) { + const dyn_type* t = dynFunction_returnType(dynFunction); return t->descriptor != 'V'; +} + +const char* dynFunction_getName(const dyn_function_type* func) { + return func->name; } \ No newline at end of file diff --git a/libs/dfi/src/dyn_function_common.h b/libs/dfi/src/dyn_function_common.h index 8d96f7cb8..68c2a844a 100644 --- a/libs/dfi/src/dyn_function_common.h +++ b/libs/dfi/src/dyn_function_common.h @@ -33,27 +33,18 @@ extern "C" { #endif struct _dyn_function_type { - char *name; - struct types_head *refTypes; //NOTE not owned - TAILQ_HEAD(,_dyn_function_argument_type) arguments; - ffi_type **ffiArguments; - dyn_type *funcReturn; + char* name; + struct types_head* refTypes; //NOTE not owned + struct dyn_function_arguments_head arguments; + ffi_type** ffiArguments; + dyn_type* funcReturn; ffi_cif cif; //closure part - ffi_closure *ffiClosure; + ffi_closure* ffiClosure; void (*fn)(void); void *userData; - void (*bind)(void *userData, void *args[], void *ret); -}; - -typedef struct _dyn_function_argument_type dyn_function_argument_type; -struct _dyn_function_argument_type { - int index; - char *name; - enum dyn_function_argument_meta argumentMeta; - dyn_type *type; - TAILQ_ENTRY(_dyn_function_argument_type) entries; + void (*bind)(void* userData, void* args[], void* ret); }; #ifdef __cplusplus diff --git a/libs/dfi/src/dyn_interface.c b/libs/dfi/src/dyn_interface.c index bad963353..5bbd78bf4 100644 --- a/libs/dfi/src/dyn_interface.c +++ b/libs/dfi/src/dyn_interface.c @@ -18,262 +18,100 @@ */ #include "dyn_interface.h" +#include "dyn_common.h" +#include "dyn_type.h" +#include "dyn_interface_common.h" +#include "celix_err.h" +#include "celix_stdlib_cleanup.h" +#include #include #include -#include "dyn_common.h" -#include "dyn_type.h" -#include "dyn_interface_common.h" -DFI_SETUP_LOG(dynInterface); static const int OK = 0; static const int ERROR = 1; -int dynInterface_checkInterface(dyn_interface_type *intf); - -static int dynInterface_parseSection(dyn_interface_type *intf, FILE *stream); -static int dynInterface_parseAnnotations(dyn_interface_type *intf, FILE *stream); -static int dynInterface_parseTypes(dyn_interface_type *intf, FILE *stream); -static int dynInterface_parseMethods(dyn_interface_type *intf, FILE *stream); -static int dynInterface_parseHeader(dyn_interface_type *intf, FILE *stream); -static int dynInterface_parseNameValueSection(dyn_interface_type *intf, FILE *stream, struct namvals_head *head); -static int dynInterface_getEntryForHead(struct namvals_head *head, const char *name, char **value); - -int dynInterface_parse(FILE *descriptor, dyn_interface_type **out) { - int status = OK; - - dyn_interface_type *intf = calloc(1, sizeof(*intf)); - if (intf != NULL) { - TAILQ_INIT(&intf->header); - TAILQ_INIT(&intf->annotations); - TAILQ_INIT(&intf->types); - TAILQ_INIT(&intf->methods); - - char peek = (char)fgetc(descriptor); - while (peek == ':') { - ungetc(peek, descriptor); - status = dynInterface_parseSection(intf, descriptor); - if (status == OK) { - peek = (char)fgetc(descriptor); - } else { - break; - } - } - - if (status == OK) { - status = dynCommon_eatChar(descriptor, EOF); - } - - if (status == OK) { - status = dynInterface_checkInterface(intf); - } - - if (status == OK) { /* We are sure that version field is present in the header */ - char* version = NULL; - dynInterface_getVersionString(intf,&version); - if (version != NULL){ - intf->version = celix_version_createVersionFromString(version); - status = intf->version != NULL ? OK : ERROR; - } - if (status == ERROR) { - LOG_ERROR("Invalid version (%s) in parsed descriptor\n",version); - } - } - } else { - status = ERROR; - LOG_ERROR("Error allocating memory for dynamic interface\n"); - } - - if (status == OK) { - *out = intf; - } else if (intf != NULL) { - dynInterface_destroy(intf); - } - return status; -} - -int dynInterface_checkInterface(dyn_interface_type *intf) { - int status = OK; - - //check header section - if (status == OK) { - bool foundType = false; - bool foundVersion = false; - bool foundName = false; - struct namval_entry *entry = NULL; - TAILQ_FOREACH(entry, &intf->header, entries) { - if (strcmp(entry->name, "type") == 0) { - foundType = true; - } else if (strcmp(entry->name, "version") == 0) { - foundVersion = true; - } else if (strcmp(entry->name, "name") == 0) { - foundName = true; - } - } - - if (!foundType || !foundVersion || !foundName) { - status = ERROR; - LOG_ERROR("Parse Error. There must be a header section with a type, version and name entry"); - } - - struct method_entry *mEntry = NULL; - TAILQ_FOREACH(mEntry, &intf->methods, entries) { - dyn_type *type = dynFunction_returnType(mEntry->dynFunc); - int descriptor = dynType_descriptorType(type); - if (descriptor != 'N') { - status = ERROR; - LOG_ERROR("Parse Error. Only method with a return type 'N' (native int) are supported. Got return type '%c'\n", descriptor); - break; - } - } - } +static int dynInterface_checkInterface(dyn_interface_type* intf); +static int dynInterface_parseSection(celix_descriptor_t* desc, const char* secName, FILE* stream); +static int dynInterface_parseMethods(dyn_interface_type* intf, FILE* stream); - return status; -} - -static int dynInterface_parseSection(dyn_interface_type *intf, FILE *stream) { +int dynInterface_parse(FILE* descriptor, dyn_interface_type** out) { int status = OK; - char *sectionName = NULL; - status = dynCommon_eatChar(stream, ':'); - - if (status == OK) { - status = dynCommon_parseName(stream, §ionName); + celix_autoptr(dyn_interface_type) intf = calloc(1, sizeof(*intf)); + if (intf == NULL) { + celix_err_pushf("Error allocating memory for dynamic interface"); + return ERROR; } - if (status == OK) { - status = dynCommon_eatChar(stream, '\n'); - } + TAILQ_INIT(&intf->header); + TAILQ_INIT(&intf->annotations); + TAILQ_INIT(&intf->types); + TAILQ_INIT(&intf->methods); - if (status == OK) { - if (strcmp("header", sectionName) == 0) { - status = dynInterface_parseHeader(intf, stream); - } else if (strcmp("annotations", sectionName) == 0) { - status = dynInterface_parseAnnotations(intf, stream); - } else if (strcmp("types", sectionName) == 0) { - status = dynInterface_parseTypes(intf, stream); - } else if (strcmp("methods", sectionName) == 0) { - status = dynInterface_parseMethods(intf, stream); - } else { - status = ERROR; - LOG_ERROR("unsupported section '%s'", sectionName); - } + if ((status = celix_dynDescriptor_parse((celix_descriptor_t*)intf, descriptor, dynInterface_parseSection)) != OK) { + return status; } - if (sectionName != NULL) { - free(sectionName); + if ((status = dynInterface_checkInterface(intf)) != OK) { + return status; } - return status; -} - -static int dynInterface_parseHeader(dyn_interface_type *intf, FILE *stream) { - return dynInterface_parseNameValueSection(intf, stream, &intf->header); -} - -static int dynInterface_parseAnnotations(dyn_interface_type *intf, FILE *stream) { - return dynInterface_parseNameValueSection(intf, stream, &intf->annotations); -} - -static int dynInterface_parseNameValueSection(dyn_interface_type *intf, FILE *stream, struct namvals_head *head) { - int status = OK; - - int peek = fgetc(stream); - while (peek != ':' && peek != EOF) { - ungetc(peek, stream); - - char *name = NULL; - char *value = NULL; - status = dynCommon_parseNameValue(stream, &name, &value); - - if (status == OK) { - status = dynCommon_eatChar(stream, '\n'); - } - - struct namval_entry *entry = NULL; - if (status == OK) { - entry = calloc(1, sizeof(*entry)); - if (entry != NULL) { - entry->name = name; - entry->value = value; - TAILQ_INSERT_TAIL(head, entry, entries); - } else { - status = ERROR; - LOG_ERROR("Error allocating memory for namval entry"); - } - } - - if (status != OK) { - if (name != NULL) { - free(name); - } - if (value != NULL) { - free(value); + *out = celix_steal_ptr(intf); + return OK; +} + +static int dynInterface_checkInterface(dyn_interface_type* intf) { + struct method_entry* mEntry = NULL; + TAILQ_FOREACH(mEntry, &intf->methods, entries) { + const dyn_type* type = dynFunction_returnType(mEntry->dynFunc); + int descriptor = dynType_descriptorType(type); + if (descriptor != 'N') { + celix_err_pushf("Parse Error. Got return type '%c' rather than 'N' (native int) for method %s (%d)", + descriptor, mEntry->id, mEntry->index); + return ERROR; + } + const struct dyn_function_arguments_head* args = dynFunction_arguments(mEntry->dynFunc); + const dyn_function_argument_type* first = TAILQ_FIRST(args); + const dyn_function_argument_type* last = TAILQ_LAST(args, dyn_function_arguments_head); + if (first == NULL || first->argumentMeta != DYN_FUNCTION_ARGUMENT_META__HANDLE) { + celix_err_pushf("Parse Error. The first argument must be handle for method %s (%d)", mEntry->id, mEntry->index); + return ERROR; + } + const dyn_function_argument_type* argEntry = NULL; + TAILQ_FOREACH(argEntry, args, entries) { + if (argEntry->argumentMeta == DYN_FUNCTION_ARGUMENT_META__HANDLE) { + if (argEntry != first) { + celix_err_pushf("Parse Error. Handle argument is only allowed as the first argument for method %s (%d)", + mEntry->id, mEntry->index); + return ERROR; + } + } else if (argEntry->argumentMeta == DYN_FUNCTION_ARGUMENT_META__OUTPUT || + argEntry->argumentMeta == DYN_FUNCTION_ARGUMENT_META__PRE_ALLOCATED_OUTPUT) { + if (argEntry != last) { + celix_err_pushf("Parse Error. Output argument is only allowed as the last argument for method %s (%d)", + mEntry->id, mEntry->index); + return ERROR; + } } - break; } - peek = fgetc(stream); } - ungetc(peek, stream); - return status; + return OK; } -static int dynInterface_parseTypes(dyn_interface_type *intf, FILE *stream) { - int status = OK; - - //expected input (Name)=\n - int peek = fgetc(stream); - while (peek != ':' && peek != EOF) { - ungetc(peek, stream); - - char *name = NULL; - status = dynCommon_parseName(stream, &name); - - if (status == OK) { - status = dynCommon_eatChar(stream, '='); - } - - dyn_type *type = NULL; - if (status == OK) { - dynType_parse(stream, name, &intf->types, &type); - } - if (name != NULL) { - free(name); - } - - if (status == OK) { - status = dynCommon_eatChar(stream, '\n'); - } - - struct type_entry *entry = NULL; - if (status == OK) { - entry = calloc(1, sizeof(*entry)); - if (entry != NULL) { - entry->type = type; - TAILQ_INSERT_TAIL(&intf->types, entry, entries); - } else { - status = ERROR; - LOG_ERROR("Error allocating memory for type entry"); - } - } - - if (status != OK) { - if (type != NULL) { - dynType_destroy(type); - } - break; - } - peek = fgetc(stream); +static int dynInterface_parseSection(celix_descriptor_t* desc, const char* secName, FILE* stream) { + dyn_interface_type* intf = (dyn_interface_type*)desc; + if (strcmp("methods", secName) != 0) { + celix_err_pushf("unsupported section '%s'", secName); + return ERROR; } - ungetc(peek, stream); - - return status; + return dynInterface_parseMethods(intf, stream); } -static int dynInterface_parseMethods(dyn_interface_type *intf, FILE *stream) { +static int dynInterface_parseMethods(dyn_interface_type* intf, FILE* stream) { int status = OK; //expected input (Name)=\n @@ -282,158 +120,105 @@ static int dynInterface_parseMethods(dyn_interface_type *intf, FILE *stream) { while (peek != ':' && peek != EOF) { ungetc(peek, stream); - char *id = NULL; - status = dynCommon_parseNameAlsoAccept(stream, ".();[{}/", &id); - - if (status == OK) { - status = dynCommon_eatChar(stream, '='); + celix_autofree char* id = NULL; + if ((status = dynCommon_parseNameAlsoAccept(stream, ".();[{}/", &id)) != OK) { + return status; } + if ((status = dynCommon_eatChar(stream, '=')) != OK) { + return status; + } - dyn_function_type *func = NULL; - if (status == OK) { - status = dynFunction_parse(stream, &intf->types, &func); + celix_autoptr(dyn_function_type) func = NULL; + if ((status = dynFunction_parse(stream, &intf->types, &func)) != OK) { + return status; } - if (status == OK) { - status = dynCommon_eatChar(stream, '\n'); + if ((status = dynCommon_eatChar(stream, '\n')) != OK) { + return status; } struct method_entry *entry = NULL; - if (status == OK) { - entry = calloc(1, sizeof(*entry)); - if (entry != NULL) { - entry->index = index++; - entry->id = id; - entry->dynFunc = func; - entry->name = strndup(id, 1024); - if (entry->name != NULL) { - int i; - for (i = 0; i < 1024; i += 1) { - if (entry->name[i] == '\0') { - break; - } else if (entry->name[i] == '(') { - entry->name[i] = '\0'; - break; - } - } - } - TAILQ_INSERT_TAIL(&intf->methods, entry, entries); - } else { - status = ERROR; - LOG_ERROR("Error allocating memory for method entry"); - } + entry = calloc(1, sizeof(*entry)); + if (entry == NULL) { + celix_err_pushf("Error allocating memory for method entry"); + return ERROR; } + entry->index = index++; + entry->id = celix_steal_ptr(id); + entry->dynFunc = celix_steal_ptr(func); + TAILQ_INSERT_TAIL(&intf->methods, entry, entries); - if (status != OK) { - if (id != NULL) { - free(id); - } - if (func != NULL) { - dynFunction_destroy(func); - //TODO free strIdentifier, name - } - break; - } peek = fgetc(stream); } - ungetc(peek, stream); + if (peek != EOF) { + ungetc(peek, stream); + } return status; } -void dynInterface_destroy(dyn_interface_type *intf) { +void dynInterface_destroy(dyn_interface_type* intf) { if (intf != NULL) { - dynCommon_clearNamValHead(&intf->header); - dynCommon_clearNamValHead(&intf->annotations); - - struct method_entry *mInfo = TAILQ_FIRST(&intf->methods); + struct method_entry* mInfo = TAILQ_FIRST(&intf->methods); while (mInfo != NULL) { - struct method_entry *mTmp = mInfo; + struct method_entry* mTmp = mInfo; mInfo = TAILQ_NEXT(mInfo, entries); if (mTmp->id != NULL) { free(mTmp->id); } - if (mTmp->name != NULL) { - free(mTmp->name); - } if (mTmp->dynFunc != NULL) { dynFunction_destroy(mTmp->dynFunc); } free(mTmp); } - - struct type_entry *tInfo = TAILQ_FIRST(&intf->types); - while (tInfo != NULL) { - struct type_entry *tmp = tInfo; - tInfo = TAILQ_NEXT(tInfo, entries); - dynType_destroy(tmp->type); - free(tmp); - } - - if(intf->version!=NULL){ - celix_version_destroy(intf->version); - } - - free(intf); - } + celix_dynDescriptor_destroy((celix_descriptor_t*)intf); + } } -int dynInterface_getName(dyn_interface_type *intf, char **out) { - return dynInterface_getEntryForHead(&intf->header, "name", out); +const char* dynInterface_getName(const dyn_interface_type* intf) { + const char* name = NULL; + // celix_dynDescriptor_checkInterface ensures that the name is present + (void)dynCommon_getEntryForHead(&intf->header, "name", &name); + return name; } -int dynInterface_getVersion(dyn_interface_type* intf , celix_version_t** version){ - *version = intf->version; - if(*version==NULL){ - return ERROR; - } - return OK; +const celix_version_t* dynInterface_getVersion(const dyn_interface_type* intf){ + // celix_dynDescriptor_checkInterface ensures that version is present + return intf->version; } -int dynInterface_getVersionString(dyn_interface_type *intf, char **version) { - return dynInterface_getEntryForHead(&intf->header, "version", version); +const char* dynInterface_getVersionString(const dyn_interface_type* intf) { + const char* version = NULL; + // celix_dynDescriptor_checkInterface ensures that the version is present + (void)dynCommon_getEntryForHead(&intf->header, "version", &version); + return version; } -int dynInterface_getHeaderEntry(dyn_interface_type *intf, const char *name, char **value) { - return dynInterface_getEntryForHead(&intf->header, name, value); +int dynInterface_getHeaderEntry(const dyn_interface_type* intf, const char* name, const char** value) { + return dynCommon_getEntryForHead(&intf->header, name, value); } -int dynInterface_getAnnotationEntry(dyn_interface_type *intf, const char *name, char **value) { - return dynInterface_getEntryForHead(&intf->annotations, name, value); +int dynInterface_getAnnotationEntry(const dyn_interface_type* intf, const char* name, const char** value) { + return dynCommon_getEntryForHead(&intf->annotations, name, value); } -static int dynInterface_getEntryForHead(struct namvals_head *head, const char *name, char **out) { - int status = OK; - char *value = NULL; - struct namval_entry *entry = NULL; - TAILQ_FOREACH(entry, head, entries) { - if (strcmp(name, entry->name) == 0) { - value = entry->value; - break; - } - } - if (value != NULL) { - *out = value; - } else { - status = ERROR; - LOG_ERROR("Cannot find '%s' in list", name); - } - return status; +const struct methods_head* dynInterface_methods(const dyn_interface_type* intf) { + return &intf->methods; } -int dynInterface_methods(dyn_interface_type *intf, struct methods_head **list) { - int status = OK; - *list = &intf->methods; - return status; +int dynInterface_nrOfMethods(const dyn_interface_type* intf) { + struct method_entry* last = TAILQ_LAST(&intf->methods, methods_head); + return last == NULL ? 0 : (last->index+1); } -int dynInterface_nrOfMethods(dyn_interface_type *intf) { - int count = 0; - struct method_entry *entry = NULL; +const struct method_entry* dynInterface_findMethod(const dyn_interface_type* intf, const char* id) { + const struct method_entry* entry = NULL; TAILQ_FOREACH(entry, &intf->methods, entries) { - count +=1; + if (strcmp(entry->id, id) == 0) { + break; + } } - return count; -} + return entry; +} \ No newline at end of file diff --git a/libs/dfi/src/dyn_interface_common.h b/libs/dfi/src/dyn_interface_common.h index b3fcd5436..e1e6d66b3 100644 --- a/libs/dfi/src/dyn_interface_common.h +++ b/libs/dfi/src/dyn_interface_common.h @@ -21,6 +21,7 @@ #define _DYN_INTERFACE_COMMON_H_ #include "dyn_interface.h" +#include "dyn_descriptor.h" #include #include @@ -33,11 +34,8 @@ extern "C" { #endif struct _dyn_interface_type { - struct namvals_head header; - struct namvals_head annotations; - struct types_head types; + CELIX_DESCRIPTOR_FIELDS struct methods_head methods; - celix_version_t* version; }; #ifdef __cplusplus diff --git a/libs/dfi/src/dyn_message.c b/libs/dfi/src/dyn_message.c index 0571be093..95cdc67fc 100644 --- a/libs/dfi/src/dyn_message.c +++ b/libs/dfi/src/dyn_message.c @@ -18,6 +18,8 @@ */ #include "dyn_message.h" +#include "dyn_descriptor.h" +#include "celix_err.h" #include #include @@ -25,335 +27,101 @@ #include "dyn_common.h" #include "dyn_type.h" -DFI_SETUP_LOG(dynMessage); - struct _dyn_message_type { - struct namvals_head header; - struct namvals_head annotations; - struct types_head types; - dyn_type *msgType; - celix_version_t* msgVersion; + CELIX_DESCRIPTOR_FIELDS + dyn_type* msgType; }; static const int OK = 0; static const int ERROR = 1; -static int dynMessage_parseSection(dyn_message_type *msg, FILE *stream); -static int dynMessage_parseAnnotations(dyn_message_type *msg, FILE *stream); -static int dynMessage_parseTypes(dyn_message_type *msg, FILE *stream); -static int dynMessage_parseMessage(dyn_message_type *msg, FILE *stream); -static int dynMessage_parseHeader(dyn_message_type *msg, FILE *stream); -static int dynMessage_parseNameValueSection(dyn_message_type *msg, FILE *stream, struct namvals_head *head); -static int dynMessage_checkMessage(dyn_message_type *msg); -static int dynMessage_getEntryForHead(struct namvals_head *head, const char *name, char **value); - -int dynMessage_parse(FILE *descriptor, dyn_message_type **out) { - int status = OK; - - dyn_message_type *msg = calloc(1, sizeof(*msg)); - if (msg != NULL) { - TAILQ_INIT(&msg->header); - TAILQ_INIT(&msg->annotations); - TAILQ_INIT(&msg->types); - - char peek = (char)fgetc(descriptor); - while (peek == ':') { - ungetc(peek, descriptor); - status = dynMessage_parseSection(msg, descriptor); - if (status == OK) { - peek = (char)fgetc(descriptor); - } else { - break; - } - } - - if (status == OK) { - status = dynCommon_eatChar(descriptor, EOF); - } - - if (status == OK) { - status = dynMessage_checkMessage(msg); - } - - if (status == OK) { /* We are sure that version field is present in the header */ - char* version=NULL; - dynMessage_getVersionString(msg,&version); - if (version != NULL) { - msg->msgVersion = celix_version_createVersionFromString(version); - status = msg->msgVersion != NULL ? OK : ERROR; - } - if (status == ERROR) { - LOG_ERROR("Invalid version (%s) in parsed descriptor\n",version); - } - } - - } else { - status = ERROR; - LOG_ERROR("Error allocating memory for dynamic message\n"); - } +static int dynMessage_parseSection(celix_descriptor_t* desc, const char* secName, FILE* stream); +static int dynMessage_parseMessage(dyn_message_type* msg, FILE* stream); - if (status == OK) { - *out = msg; - } else if (msg != NULL) { - LOG_ERROR("Error parsing msg\n"); - dynMessage_destroy(msg); - } - return status; -} - -static int dynMessage_checkMessage(dyn_message_type *msg) { +int dynMessage_parse(FILE* descriptor, dyn_message_type** out) { int status = OK; - //check header section - if (status == OK) { - bool foundType = false; - bool foundVersion = false; - bool foundName = false; - struct namval_entry *entry = NULL; - TAILQ_FOREACH(entry, &msg->header, entries) { - if (strcmp(entry->name, "type") == 0) { - foundType = true; - } else if (strcmp(entry->name, "version") == 0) { - foundVersion = true; - } else if (strcmp(entry->name, "name") == 0) { - foundName = true; - } - } - - if (!foundType || !foundVersion || !foundName) { - status = ERROR; - LOG_ERROR("Parse Error. There must be a header section with a type, version and name entry"); - } + celix_autoptr(dyn_message_type) msg = calloc(1, sizeof(*msg)); + if (msg == NULL) { + celix_err_push("Error allocating memory for dynamic message"); + return ERROR; } + TAILQ_INIT(&msg->header); + TAILQ_INIT(&msg->annotations); + TAILQ_INIT(&msg->types); - return status; -} - -static int dynMessage_parseSection(dyn_message_type *msg, FILE *stream) { - int status; - char *sectionName = NULL; - - status = dynCommon_eatChar(stream, ':'); - - if (status == OK) { - status = dynCommon_parseName(stream, §ionName); - } - - if (status == OK) { - status = dynCommon_eatChar(stream, '\n'); - } - - if (status == OK) { - if (strcmp("header", sectionName) == 0) { - status = dynMessage_parseHeader(msg, stream); - } else if (strcmp("annotations", sectionName) == 0) { - status = dynMessage_parseAnnotations(msg, stream); - } else if (strcmp("types", sectionName) == 0) { - status = dynMessage_parseTypes(msg, stream); - } else if (strcmp("message", sectionName) == 0) { - status = dynMessage_parseMessage(msg, stream); - } else { - status = ERROR; - LOG_ERROR("unsupported section '%s'", sectionName); - } - } - - if (sectionName != NULL) { - free(sectionName); - } - - return status; -} - -static int dynMessage_parseHeader(dyn_message_type *msg, FILE *stream) { - return dynMessage_parseNameValueSection(msg, stream, &msg->header); -} - -static int dynMessage_parseAnnotations(dyn_message_type *msg, FILE *stream) { - return dynMessage_parseNameValueSection(msg, stream, &msg->annotations); -} - -static int dynMessage_parseNameValueSection(dyn_message_type *msg, FILE *stream, struct namvals_head *head) { - int status = OK; - - int peek = fgetc(stream); - while (peek != ':' && peek != EOF) { - ungetc(peek, stream); - - char *name = NULL; - char *value = NULL; - status = dynCommon_parseNameValue(stream, &name, &value); - - if (status == OK) { - status = dynCommon_eatChar(stream, '\n'); - } - - struct namval_entry *entry = NULL; - if (status == OK) { - entry = calloc(1, sizeof(*entry)); - if (entry != NULL) { - entry->name = name; - entry->value = value; - TAILQ_INSERT_TAIL(head, entry, entries); - } else { - status = ERROR; - LOG_ERROR("Error allocating memory for namval entry"); - } - } - - if (status != OK) { - if (name != NULL) { - free(name); - } - if (value != NULL) { - free(value); - } - break; - } - peek = fgetc(stream); + if ((status = celix_dynDescriptor_parse((celix_descriptor_t*)msg, descriptor, dynMessage_parseSection)) != OK) { + return status; } - ungetc(peek, stream); + *out = celix_steal_ptr(msg); return status; } -static int dynMessage_parseTypes(dyn_message_type *msg, FILE *stream) { - int status = OK; - - //expected input (Name)=\n - int peek = fgetc(stream); - while (peek != ':' && peek != EOF) { - ungetc(peek, stream); - - char *name = NULL; - status = dynCommon_parseName(stream, &name); - - if (status == OK) { - status = dynCommon_eatChar(stream, '='); - } - - dyn_type *type = NULL; - if (status == OK) { - status = dynType_parse(stream, name, &msg->types, &type); - } - - if (status == OK) { - status = dynCommon_eatChar(stream, '\n'); - } - - struct type_entry *entry = NULL; - if (status == OK) { - entry = calloc(1, sizeof(*entry)); - if (entry != NULL) { - entry->type = type; - TAILQ_INSERT_TAIL(&msg->types, entry, entries); - } else { - status = ERROR; - LOG_ERROR("Error allocating memory for type entry"); - } - } - - if (name != NULL) { - free(name); - } - - if (status != OK) { - if (type != NULL) { - dynType_destroy(type); - } - break; - } - peek = fgetc(stream); +static int dynMessage_parseSection(celix_descriptor_t* desc, const char* secName, FILE* stream) { + dyn_message_type* msg = (dyn_message_type*)desc; + if (strcmp("message", secName) != 0) { + celix_err_pushf("unsupported section '%s'", secName); + return ERROR; } - ungetc(peek, stream); - - return status; + return dynMessage_parseMessage(msg, stream); } -static int dynMessage_parseMessage(dyn_message_type *msg, FILE *stream) { +static int dynMessage_parseMessage(dyn_message_type* msg, FILE* stream) { int status; //expected input \n - char *name = NULL; - status = dynMessage_getName(msg, &name); + const char* name = NULL; + if ((status = dynCommon_getEntryForHead(&msg->header, "name", &name)) != OK) { + return status; + } - if (status == OK) { - status = dynType_parse(stream, name, &(msg->types), &(msg->msgType)); + if ((status = dynType_parse(stream, name, &(msg->types), &(msg->msgType))) != OK) { + return status; } + if ((status = dynCommon_eatChar(stream, '\n')) != OK) { + return status; + } return status; } -void dynMessage_destroy(dyn_message_type *msg) { +void dynMessage_destroy(dyn_message_type* msg) { if (msg != NULL) { - dynCommon_clearNamValHead(&msg->header); - dynCommon_clearNamValHead(&msg->annotations); - - struct type_entry *tInfo = TAILQ_FIRST(&msg->types); - while (tInfo != NULL) { - struct type_entry *tmp = tInfo; - tInfo = TAILQ_NEXT(tInfo, entries); - dynType_destroy(tmp->type); - free(tmp); - } - if (msg->msgType != NULL) { - dynType_destroy(msg->msgType); + dynType_destroy(msg->msgType); } - - if(msg->msgVersion != NULL){ - celix_version_destroy(msg->msgVersion); - } - - free(msg); - } -} - -int dynMessage_getName(dyn_message_type *msg, char **out) { - return dynMessage_getEntryForHead(&msg->header, "name", out); + celix_dynDescriptor_destroy((celix_descriptor_t*)msg); + } } -int dynMessage_getVersion(dyn_message_type *msg, celix_version_t** version){ - *version = msg->msgVersion; - if(*version==NULL){ - return ERROR; - } - return OK; +const char* dynMessage_getName(const dyn_message_type* msg) { + const char* name = NULL; + // celix_dynDescriptor_checkInterface ensures that the name is present + (void)dynCommon_getEntryForHead(&msg->header, "name", &name); + return name; } -int dynMessage_getVersionString(dyn_message_type *msg, char **version) { - return dynMessage_getEntryForHead(&msg->header, "version", version); +const celix_version_t* dynMessage_getVersion(const dyn_message_type* msg){ + // celix_dynDescriptor_checkInterface ensures that version is present + return msg->version; } -int dynMessage_getHeaderEntry(dyn_message_type *msg, const char *name, char **value) { - return dynMessage_getEntryForHead(&msg->header, name, value); +const char* dynMessage_getVersionString(const dyn_message_type* msg) { + const char* version = NULL; + // celix_dynDescriptor_checkInterface ensures that the version is present + (void)dynCommon_getEntryForHead(&msg->header, "version", &version); + return version; } -int dynMessage_getAnnotationEntry(dyn_message_type *msg, const char *name, char **value) { - return dynMessage_getEntryForHead(&msg->annotations, name, value); +int dynMessage_getHeaderEntry(dyn_message_type* msg, const char* name, const char** value) { + return dynCommon_getEntryForHead(&msg->header, name, value); } -static int dynMessage_getEntryForHead(struct namvals_head *head, const char *name, char **out) { - int status = OK; - char *value = NULL; - struct namval_entry *entry = NULL; - TAILQ_FOREACH(entry, head, entries) { - if (strcmp(name, entry->name) == 0) { - value = entry->value; - break; - } - } - if (value != NULL) { - *out = value; - } else { - status = ERROR; - LOG_ERROR("Cannot find '%s' in list", name); - } - return status; +int dynMessage_getAnnotationEntry(dyn_message_type* msg, const char* name, const char** value) { + return dynCommon_getEntryForHead(&msg->annotations, name, value); } -int dynMessage_getMessageType(dyn_message_type *msg, dyn_type **type) { - int status = OK; - *type = msg->msgType; - return status; +const dyn_type* dynMessage_getMessageType(dyn_message_type* msg) { + return msg->msgType; } diff --git a/libs/dfi/src/dyn_type.c b/libs/dfi/src/dyn_type.c index df2021bfd..399bb7663 100644 --- a/libs/dfi/src/dyn_type.c +++ b/libs/dfi/src/dyn_type.c @@ -18,128 +18,130 @@ */ #include "dyn_type.h" +#include "dyn_type_common.h" +#include "dyn_common.h" +#include "celix_err.h" +#include "celix_stdio_cleanup.h" +#include "celix_stdlib_cleanup.h" #include #include #include #include #include -#include "dyn_type_common.h" - -#include "dyn_type_common.h" -#include "dyn_common.h" - -DFI_SETUP_LOG(dynType) static const int OK = 0; static const int ERROR = 1; static const int MEM_ERROR = 2; static const int PARSE_ERROR = 3; -static int dynType_parseWithStream(FILE *stream, const char *name, dyn_type *parent, struct types_head *refTypes, dyn_type **result); -static void dynType_clear(dyn_type *type); -static void dynType_clearComplex(dyn_type *type); -static void dynType_clearSequence(dyn_type *type); -static void dynType_clearTypedPointer(dyn_type *type); -ffi_type * dynType_ffiType(dyn_type *type); - -static struct type_entry *dynType_allocTypeEntry(void); - -static ffi_type * dynType_ffiTypeFor(int c); -static int dynType_parseAny(FILE *stream, dyn_type *type); -static int dynType_parseComplex(FILE *stream, dyn_type *type); -static int dynType_parseNestedType(FILE *stream, dyn_type *type); -static int dynType_parseReference(FILE *stream, dyn_type *type); -static int dynType_parseRefByValue(FILE *stream, dyn_type *type); -static int dynType_parseSequence(FILE *stream, dyn_type *type); -static int dynType_parseSimple(int c, dyn_type *type); -static int dynType_parseTypedPointer(FILE *stream, dyn_type *type); -static unsigned short dynType_getOffset(dyn_type *type, int index); - -static void dynType_printAny(char *name, dyn_type *type, int depth, FILE *stream); -static void dynType_printComplex(char *name, dyn_type *type, int depth, FILE *stream); -static void dynType_printSequence(char *name, dyn_type *type, int depth, FILE *stream); -static void dynType_printSimple(char *name, dyn_type *type, int depth, FILE *stream); -static void dynType_printEnum(char *name, dyn_type *type, int depth, FILE *stream); -static void dynType_printTypedPointer(char *name, dyn_type *type, int depth, FILE *stream); -static void dynType_printDepth(int depth, FILE *stream); - -static void dynType_printTypes(dyn_type *type, FILE *stream); -static void dynType_printComplexType(dyn_type *type, FILE *stream); -static void dynType_printSimpleType(dyn_type *type, FILE *stream); - -static int dynType_parseText(FILE *stream, dyn_type *type); -static int dynType_parseEnum(FILE *stream, dyn_type *type); -void dynType_freeComplexType(dyn_type *type, void *loc); -void dynType_deepFree(dyn_type *type, void *loc, bool alsoDeleteSelf); -void dynType_freeSequenceType(dyn_type *type, void *seqLoc); - -static int dynType_parseMetaInfo(FILE *stream, dyn_type *type); +static int dynType_parseWithStream(FILE* stream, const char* name, dyn_type* parent, const struct types_head* refTypes, dyn_type** result); +static int dynType_parseWithStreamOfName(FILE* stream, char* name, dyn_type* parent, const struct types_head* refTypes, dyn_type** result, int (*)(FILE*, dyn_type*)); +static void dynType_clear(dyn_type* type); +static void dynType_clearComplex(dyn_type* type); +static void dynType_clearSequence(dyn_type* type); +static void dynType_clearTypedPointer(dyn_type* type); + +static ffi_type* dynType_ffiTypeFor(int c); +static int dynType_parseAny(FILE* stream, dyn_type* type); +static int dynType_parseComplex(FILE* stream, dyn_type* type); +static int dynType_parseNestedType(FILE* stream, dyn_type* type); +static int dynType_parseReference(FILE* stream, dyn_type* type); +static int dynType_parseRefByValue(FILE* stream, dyn_type* type); +static int dynType_parseSequence(FILE* stream, dyn_type* type); +static int dynType_parseSimple(int c, dyn_type* type); +static int dynType_parseTypedPointer(FILE* stream, dyn_type* type); +static unsigned short dynType_getOffset(const dyn_type* type, int index); + +static void dynType_printAny(const char* name, const dyn_type* type, int depth, FILE* stream); +static void dynType_printComplex(const char* name, const dyn_type* type, int depth, FILE* stream); +static void dynType_printSequence(const char* name, const dyn_type* type, int depth, FILE* stream); +static void dynType_printSimple(const char* name, const dyn_type* type, int depth, FILE* stream); +static void dynType_printEnum(const char* name, const dyn_type* type, int depth, FILE* stream); +static void dynType_printTypedPointer(const char* name, const dyn_type* type, int depth, FILE* stream); +static void dynType_printText(const char* name, const dyn_type* type, int depth, FILE* stream); +static void dynType_printDepth(int depth, FILE* stream); + +static void dynType_printTypes(const dyn_type* type, FILE* stream); +static void dynType_printComplexType(const dyn_type* type, FILE* stream); +static void dynType_printSimpleType(const dyn_type* type, FILE* stream); + +static int dynType_parseText(FILE* stream, dyn_type* type); +static int dynType_parseEnum(FILE* stream, dyn_type* type); +static void dynType_freeComplexType(const dyn_type* type, void* loc); +static void dynType_deepFree(const dyn_type* type, void* loc, bool alsoDeleteSelf); +static void dynType_freeSequenceType(const dyn_type* type, void* seqLoc); + +static int dynType_parseMetaInfo(FILE* stream, dyn_type* type); struct generic_sequence { uint32_t cap; uint32_t len; - void *buf; + void* buf; }; -int dynType_parse(FILE *descriptorStream, const char *name, struct types_head *refTypes, dyn_type **type) { +int dynType_parse(FILE* descriptorStream, const char* name, const struct types_head* refTypes, dyn_type** type) { return dynType_parseWithStream(descriptorStream, name, NULL, refTypes, type); } -int dynType_parseWithStr(const char *descriptor, const char *name, struct types_head *refTypes, dyn_type **type) { - int status = OK; - FILE *stream = fmemopen((char *)descriptor, strlen(descriptor) + 1, "r"); - if (stream != NULL) { - status = dynType_parseWithStream(stream, name, NULL, refTypes, type); - if (status == OK) { - int c = fgetc(stream); - if (c != '\0' && c != EOF) { - status = PARSE_ERROR; - dynType_destroy(*type); - *type = NULL; - LOG_ERROR("Expected EOF got %c", c); - } - } - fclose(stream); - } else { - status = ERROR; - LOG_ERROR("Error creating mem stream for descriptor string. %s", strerror(errno)); + +int dynType_parseOfName(FILE* descriptorStream, char* name, const struct types_head* refTypes, dyn_type** type) { + return dynType_parseWithStreamOfName(descriptorStream, name, NULL, refTypes, type, dynType_parseAny); +} + +int dynType_parseWithStr(const char* descriptor, const char* name, const struct types_head* refTypes, dyn_type** type) { + int status; + celix_autoptr(FILE) stream = fmemopen((char *)descriptor, strlen(descriptor), "r"); + if (stream == NULL) { + celix_err_pushf("Error creating mem stream for descriptor string. %s", strerror(errno)); + return ERROR; } - return status; + celix_autoptr(dyn_type) result = NULL; + if ((status = dynType_parseWithStream(stream, name, NULL, refTypes, &result)) != OK) { + return status; + } + if (dynCommon_eatChar(stream, EOF) != 0) { + return PARSE_ERROR; + } + *type = celix_steal_ptr(result); + return OK; } -static int dynType_parseWithStream(FILE *stream, const char *name, dyn_type *parent, struct types_head *refTypes, dyn_type **result) { - int status = OK; - dyn_type *type = calloc(1, sizeof(*type)); - if (type != NULL) { - type->parent = parent; - type->type = DYN_TYPE_INVALID; - type->referenceTypes = refTypes; - TAILQ_INIT(&type->nestedTypesHead); - TAILQ_INIT(&type->metaProperties); - if (name != NULL) { - type->name = strdup(name); - if (type->name == NULL) { - status = MEM_ERROR; - LOG_ERROR("Error strdup'ing name '%s'\n", name); - } - } - if (status == OK) { - status = dynType_parseAny(stream, type); - } - if (status == OK) { - *result = type; - } else { - dynType_destroy(type); +static int dynType_parseWithStream(FILE* stream, const char* name, dyn_type* parent, const struct types_head* refTypes, dyn_type** result) { + char* typeName = NULL; + if (name != NULL) { + typeName = strdup(name); + if (typeName == NULL) { + celix_err_pushf("Error strdup'ing name '%s'", name); + return MEM_ERROR; } - } else { - status = MEM_ERROR; - LOG_ERROR("Error allocating memory for type"); } - return status; + return dynType_parseWithStreamOfName(stream, typeName, parent, refTypes, result, dynType_parseAny); +} + +static int dynType_parseWithStreamOfName(FILE* stream, char* name, dyn_type* parent, const struct types_head* refTypes, + dyn_type** result, int (*parse)(FILE*, dyn_type*)) { + int status; + celix_autofree char* typeName = name; + celix_autoptr(dyn_type) type = calloc(1, sizeof(*type)); + if (type == NULL) { + celix_err_push("Error allocating memory for type"); + return MEM_ERROR; + } + type->parent = parent; + type->type = DYN_TYPE_INVALID; + type->referenceTypes = refTypes; + TAILQ_INIT(&type->nestedTypesHead); + TAILQ_INIT(&type->metaProperties); + type->name = celix_steal_ptr(typeName); + if ((status = parse(stream, type)) != OK) { + return status; + } + *result = celix_steal_ptr(type); + return OK; } -static int dynType_parseAny(FILE *stream, dyn_type *type) { +static int dynType_parseAny(FILE* stream, dyn_type* type) { int status = OK; int c = fgetc(stream); @@ -175,6 +177,8 @@ static int dynType_parseAny(FILE *stream, dyn_type *type) { status = dynType_parseMetaInfo(stream, type); if (status == OK) { status = dynType_parseAny(stream, type); + } else { + celix_err_push("Failed to parse meta properties"); } break; default : @@ -185,265 +189,190 @@ static int dynType_parseAny(FILE *stream, dyn_type *type) { return status; } -static int dynType_parseMetaInfo(FILE *stream, dyn_type *type) { - int status = OK; - char *name = NULL; - char *value = NULL; - - struct meta_entry *entry = calloc(1, sizeof(*entry)); - if (entry == NULL) { - status = ERROR; - } +static int dynType_parseMetaInfo(FILE* stream, dyn_type* type) { + celix_autofree char* name = NULL; + celix_autofree char* value = NULL; - if (status == OK) { - status = dynCommon_parseName(stream, &name); + if (dynCommon_parseName(stream, &name) != OK) { + return PARSE_ERROR; } - - if (status == OK) { - status = dynCommon_eatChar(stream, '='); + if (dynCommon_eatChar(stream, '=') != OK) { + return PARSE_ERROR; } - - if (status == OK) { - status = dynCommon_parseName(stream, &value); + if (dynCommon_parseName(stream, &value) != OK) { + return PARSE_ERROR; } - - if (status == OK) { - status = dynCommon_eatChar(stream, ';'); + if (dynCommon_eatChar(stream, ';') != OK) { + return PARSE_ERROR; } - - if (status == OK) { - entry->name = name; - entry->value = value; - TAILQ_INSERT_TAIL(&type->metaProperties, entry, entries); - } else { - LOG_ERROR("Failed to parse meta properties"); - free(name); - free(value); - free(entry); + struct meta_entry *entry = calloc(1, sizeof(*entry)); + if (entry == NULL) { + return MEM_ERROR; } - - return status; + entry->name = celix_steal_ptr(name); + entry->value = celix_steal_ptr(value); + TAILQ_INSERT_TAIL(&type->metaProperties, entry, entries); + return OK; } -static int dynType_parseText(FILE *stream, dyn_type *type) { - int status = OK; +static int dynType_parseText(FILE* stream, dyn_type* type) { type->type = DYN_TYPE_TEXT; type->descriptor = 't'; + type->trivial = false; type->ffiType = &ffi_type_pointer; - return status; + return OK; } -static int dynType_parseEnum(FILE *stream, dyn_type *type) { - int status = OK; +static int dynType_parseEnum(FILE* stream, dyn_type* type) { type->ffiType = &ffi_type_sint32; type->descriptor = 'E'; + type->trivial = true; type->type = DYN_TYPE_SIMPLE; - return status; + return OK; } -static int dynType_parseComplex(FILE *stream, dyn_type *type) { +static int dynType_parseComplex(FILE* stream, dyn_type* type) { + size_t nbEntries = 0; int status = OK; type->type = DYN_TYPE_COMPLEX; type->descriptor = '{'; + type->trivial = true; type->ffiType = &type->complex.structType; TAILQ_INIT(&type->complex.entriesHead); int c = fgetc(stream); - struct complex_type_entry *entry = NULL; + struct complex_type_entry* entry = NULL; while (c != ' ' && c != '}') { ungetc(c,stream); + celix_autoptr(dyn_type) subType = NULL; + status = dynType_parseWithStreamOfName(stream, NULL, type, NULL, &subType, dynType_parseAny); + if (status != OK) { + return status; + } entry = calloc(1, sizeof(*entry)); - if (entry != NULL) { - entry->type = calloc(1, sizeof(*entry->type)); + if (entry == NULL) { + celix_err_push("Error allocating memory for complex_type_entry"); + return MEM_ERROR; } - if (entry != NULL && entry->type != NULL) { - entry->type->parent = type; - entry->type->type = DYN_TYPE_INVALID; - TAILQ_INIT(&entry->type->nestedTypesHead); - TAILQ_INIT(&entry->type->metaProperties); - TAILQ_INSERT_TAIL(&type->complex.entriesHead, entry, entries); - status = dynType_parseAny(stream, entry->type); - } else { - free(entry); - status = MEM_ERROR; - LOG_ERROR("Error allocating memory for type"); + entry->type = celix_steal_ptr(subType); + if (!dynType_isTrivial(entry->type)) { + type->trivial = false; } - - if (status != OK) { - break; - } - + TAILQ_INSERT_TAIL(&type->complex.entriesHead, entry, entries); + nbEntries += 1; c = fgetc(stream); } -// loop over names - if (status == OK) { - entry = TAILQ_FIRST(&type->complex.entriesHead); - char *name = NULL; - while (c == ' ' && entry != NULL) { - status = dynCommon_parseName(stream, &name); - if (status == OK) { - entry->name = name; - entry = TAILQ_NEXT(entry, entries); - } else { - break; - } - c = getc(stream); + // loop over names + entry = TAILQ_FIRST(&type->complex.entriesHead); + char* name = NULL; + // the current implementation permits trailing unnamed fields, i.e. number of names is less than number of fields + while (c == ' ' && entry != NULL) { + if ((status = dynCommon_parseName(stream, &name)) != OK) { + return status; } + entry->name = name; + entry = TAILQ_NEXT(entry, entries); + c = getc(stream); } - - int count = 0; - if (status == OK) { - TAILQ_FOREACH(entry, &type->complex.entriesHead, entries) { - count +=1; - } + if (c != '}') { + celix_err_push("Error parsing complex type, expected '}'"); + return PARSE_ERROR; } - if (status == OK) { - type->complex.structType.type = FFI_TYPE_STRUCT; - type->complex.structType.elements = calloc(count + 1, sizeof(ffi_type*)); - if (type->complex.structType.elements != NULL) { - type->complex.structType.elements[count] = NULL; - int index = 0; - TAILQ_FOREACH(entry, &type->complex.entriesHead, entries) { - type->complex.structType.elements[index++] = dynType_ffiType(entry->type); - } - } else { - status = MEM_ERROR; - LOG_ERROR("Error allocating memory for elements"); - } + type->complex.structType.type = FFI_TYPE_STRUCT; + type->complex.structType.elements = calloc(nbEntries + 1, sizeof(ffi_type*)); + if (type->complex.structType.elements == NULL) { + celix_err_push("Error allocating memory for ffi_type elements"); + return MEM_ERROR; } - - if (status == OK) { - type->complex.types = calloc(count, sizeof(dyn_type *)); - if (type->complex.types != NULL) { - int index = 0; - TAILQ_FOREACH(entry, &type->complex.entriesHead, entries) { - type->complex.types[index++] = entry->type; - } - } else { - status = MEM_ERROR; - LOG_ERROR("Error allocating memory for type"); - } + type->complex.structType.elements[nbEntries] = NULL; + int index = 0; + TAILQ_FOREACH(entry, &type->complex.entriesHead, entries) { + type->complex.structType.elements[index++] = dynType_ffiType(entry->type); } - if (status == OK) { - dynType_prepCif(type->ffiType); + type->complex.types = calloc(nbEntries, sizeof(dyn_type *)); + if (type->complex.types == NULL) { + celix_err_pushf("Error allocating memory for complex types"); + return MEM_ERROR; + } + index = 0; + TAILQ_FOREACH(entry, &type->complex.entriesHead, entries) { + type->complex.types[index++] = entry->type; } + (void)ffi_get_struct_offsets(FFI_DEFAULT_ABI, type->ffiType, NULL); return status; } -static int dynType_parseNestedType(FILE *stream, dyn_type *type) { +static int dynType_parseNestedType(FILE* stream, dyn_type* type) { int status = OK; - char *name = NULL; - struct type_entry *entry = NULL; - - entry = dynType_allocTypeEntry(); - if (entry != NULL) { - entry->type->parent = type; - entry->type->type = DYN_TYPE_INVALID; - TAILQ_INIT(&entry->type->nestedTypesHead); - TAILQ_INIT(&entry->type->metaProperties); - TAILQ_INSERT_TAIL(&type->nestedTypesHead, entry, entries); - status = dynCommon_parseName(stream, &name); - entry->type->name = name; - } else { - status = MEM_ERROR; - LOG_ERROR("Error allocating entry"); - } - - if (status == OK) { - int c = fgetc(stream); - if (c != '=') { - status = PARSE_ERROR; - LOG_ERROR("Error parsing nested type expected '=' got '%c'", c); - } + celix_autofree char* name = NULL; + celix_autoptr(dyn_type) subType = NULL; + if ((status = dynCommon_parseName(stream, &name)) != OK) { + return status; } - - if (status == OK) { - status = dynType_parseAny(stream, entry->type); - int c = fgetc(stream); - if (c != ';') { - status = PARSE_ERROR; - LOG_ERROR("Expected ';' got '%c'\n", c); - } + if (dynCommon_eatChar(stream, '=') != OK) { + return PARSE_ERROR; } - - return status; + if ((status = dynType_parseWithStreamOfName(stream, celix_steal_ptr(name), type, NULL, &subType, dynType_parseAny)) != OK) { + return status; + } + if (dynCommon_eatChar(stream, ';') != OK) { + return PARSE_ERROR; + } + struct type_entry* entry = calloc(1, sizeof(*entry)); + if (entry == NULL) { + celix_err_pushf("Error allocating entry"); + return MEM_ERROR; + } + entry->type = celix_steal_ptr(subType); + TAILQ_INSERT_TAIL(&type->nestedTypesHead, entry, entries); + return OK; } -static int dynType_parseReference(FILE *stream, dyn_type *type) { - int status = OK; +static int dynType_parseReference(FILE* stream, dyn_type* type) { + int status; type->type = DYN_TYPE_TYPED_POINTER; type->descriptor = '*'; - + type->trivial = false; type->ffiType = &ffi_type_pointer; type->typedPointer.typedType = NULL; - dyn_type *subType = calloc(1, sizeof(*subType)); - - if (subType != NULL) { - type->typedPointer.typedType = subType; - subType->parent = type; - subType->type = DYN_TYPE_INVALID; - TAILQ_INIT(&subType->nestedTypesHead); - TAILQ_INIT(&subType->metaProperties); - status = dynType_parseRefByValue(stream, subType); - } else { - status = MEM_ERROR; - LOG_ERROR("Error allocating memory for subtype\n"); + if ((status = dynType_parseWithStreamOfName(stream, NULL, type, NULL, + &type->typedPointer.typedType, dynType_parseRefByValue)) != OK) { + return status; } - - return status; + return OK; } -static int dynType_parseRefByValue(FILE *stream, dyn_type *type) { +static int dynType_parseRefByValue(FILE* stream, dyn_type* type) { int status = OK; type->type = DYN_TYPE_REF; type->descriptor = 'l'; - char *name = NULL; - status = dynCommon_parseName(stream, &name); - if (status == OK) { - dyn_type *ref = dynType_findType(type, name); - if (ref != NULL) { - type->ref.ref = ref; - } else { - status = PARSE_ERROR; - LOG_ERROR("Error cannot find type '%s'", name); - } - free(name); - } - - if (status ==OK) { - int c = fgetc(stream); - if (c != ';') { - status = PARSE_ERROR; - LOG_ERROR("Error expected ';' got '%c'", c); - } + celix_autofree char* name = NULL; + if ((status = dynCommon_parseName(stream, &name)) != OK) { + return status; } - - return status; -} - -static struct type_entry *dynType_allocTypeEntry(void) { - struct type_entry *entry = calloc(1, sizeof(*entry)); - if (entry != NULL) { - entry->type = calloc(1, sizeof(*entry->type)); - if (entry->type == NULL) { - free(entry); - entry = NULL; - } + dyn_type* ref = dynType_findType(type, name); + if (ref == NULL) { + celix_err_pushf("Error cannot find type '%s'", name); + return PARSE_ERROR; + } + type->ref.ref = ref; + if (dynCommon_eatChar(stream, ';') != OK) { + return PARSE_ERROR; } - return entry; + return OK; } -static ffi_type *seq_types[] = {&ffi_type_uint32, &ffi_type_uint32, &ffi_type_pointer, NULL}; +static ffi_type* seq_types[] = {&ffi_type_uint32, &ffi_type_uint32, &ffi_type_pointer, NULL}; -static int dynType_parseSequence(FILE *stream, dyn_type *type) { - int status = OK; +static int dynType_parseSequence(FILE* stream, dyn_type* type) { + int status; type->type = DYN_TYPE_SEQUENCE; type->descriptor = '['; @@ -452,64 +381,60 @@ static int dynType_parseSequence(FILE *stream, dyn_type *type) { type->sequence.seqType.size = 0; type->sequence.seqType.alignment = 0; - status = dynType_parseWithStream(stream, NULL, type, NULL, &type->sequence.itemType); - - if (status == OK) { - type->ffiType = &type->sequence.seqType; - dynType_prepCif(&type->sequence.seqType); + status = dynType_parseWithStreamOfName(stream, NULL, type, NULL, &type->sequence.itemType, dynType_parseAny); + if (status != OK) { + return status; } - return status; + type->ffiType = &type->sequence.seqType; + (void)ffi_get_struct_offsets(FFI_DEFAULT_ABI, type->ffiType, NULL); + return OK; } -static int dynType_parseSimple(int c, dyn_type *type) { - int status = OK; - ffi_type *ffiType = dynType_ffiTypeFor(c); - if (ffiType != NULL) { - type->type = DYN_TYPE_SIMPLE; - type->descriptor = c; - type->ffiType = ffiType; - } else { - status = PARSE_ERROR; - LOG_ERROR("Error unsupported type '%c'", c); +static int dynType_parseSimple(int c, dyn_type* type) { + ffi_type* ffiType = dynType_ffiTypeFor(c); + if (ffiType == NULL) { + celix_err_pushf("Error unsupported type '%c'", c); + return PARSE_ERROR; } - - return status; + type->type = DYN_TYPE_SIMPLE; + type->descriptor = c; + type->trivial = c != 'P'; + type->ffiType = ffiType; + return OK; } -static int dynType_parseTypedPointer(FILE *stream, dyn_type *type) { +static int dynType_parseTypedPointer(FILE* stream, dyn_type* type) { int status = OK; type->type = DYN_TYPE_TYPED_POINTER; type->descriptor = '*'; + type->trivial = false; type->ffiType = &ffi_type_pointer; - status = dynType_parseWithStream(stream, NULL, type, NULL, &type->typedPointer.typedType); + status = dynType_parseWithStreamOfName(stream, NULL, type, NULL, &type->typedPointer.typedType, dynType_parseAny); return status; } -void dynType_destroy(dyn_type *type) { +void dynType_destroy(dyn_type* type) { if (type != NULL) { dynType_clear(type); free(type); } } -static void dynType_clear(dyn_type *type) { - struct type_entry *entry = TAILQ_FIRST(&type->nestedTypesHead); - struct type_entry *tmp = NULL; +static void dynType_clear(dyn_type* type) { + struct type_entry* entry = TAILQ_FIRST(&type->nestedTypesHead); + struct type_entry* tmp = NULL; while (entry != NULL) { tmp = entry; entry = TAILQ_NEXT(entry, entries); - if (tmp->type != NULL) { - dynType_destroy(tmp->type); - tmp->type = NULL; - } + dynType_destroy(tmp->type); free(tmp); } - struct meta_entry *mEntry = TAILQ_FIRST(&type->metaProperties);; - struct meta_entry *next = NULL; + struct meta_entry* mEntry = TAILQ_FIRST(&type->metaProperties);; + struct meta_entry* next = NULL; while (mEntry != NULL) { next = TAILQ_NEXT(mEntry, entries); if (mEntry != NULL) { @@ -537,10 +462,10 @@ static void dynType_clear(dyn_type *type) { } } -static void dynType_clearComplex(dyn_type *type) { +static void dynType_clearComplex(dyn_type* type) { assert(type->type == DYN_TYPE_COMPLEX); - struct complex_type_entry *entry = TAILQ_FIRST(&type->complex.entriesHead); - struct complex_type_entry *tmp = NULL; + struct complex_type_entry* entry = TAILQ_FIRST(&type->complex.entriesHead); + struct complex_type_entry* tmp = NULL; while (entry != NULL) { dynType_destroy(entry->type); if (entry->name != NULL) { @@ -558,47 +483,42 @@ static void dynType_clearComplex(dyn_type *type) { } } -static void dynType_clearSequence(dyn_type *type) { +static void dynType_clearSequence(dyn_type* type) { assert(type->type == DYN_TYPE_SEQUENCE); if (type->sequence.itemType != NULL) { dynType_destroy(type->sequence.itemType); } } -static void dynType_clearTypedPointer(dyn_type *type) { +static void dynType_clearTypedPointer(dyn_type* type) { assert(type->type == DYN_TYPE_TYPED_POINTER); if (type->typedPointer.typedType != NULL) { dynType_destroy(type->typedPointer.typedType); } } -int dynType_alloc(dyn_type *type, void **bufLoc) { - int status = OK; +int dynType_alloc(const dyn_type* type, void** bufLoc) { + const dyn_type* current = dynType_realType(type); - if (type->type == DYN_TYPE_REF) { - status = dynType_alloc(type->ref.ref, bufLoc); - } else { - void *inst = calloc(1, type->ffiType->size); - if (inst != NULL) { - *bufLoc = inst; - } else { - status = MEM_ERROR; - LOG_ERROR("Error allocating memory for type '%c'", type->descriptor); - } + void* inst = calloc(1, current->ffiType->size); + if (inst == NULL) { + celix_err_pushf("Error allocating memory for type '%c'", current->descriptor); + return MEM_ERROR; } + *bufLoc = inst; - return status; + return OK; } -int dynType_complex_indexForName(dyn_type *type, const char *name) { +int dynType_complex_indexForName(const dyn_type* type, const char* name) { assert(type->type == DYN_TYPE_COMPLEX); if (name == NULL) { return -1; } int i = 0; int index = -1; - struct complex_type_entry *entry = NULL; + struct complex_type_entry* entry = NULL; TAILQ_FOREACH(entry, &type->complex.entriesHead, entries) { if (entry->name != NULL && strcmp(name, entry->name) == 0) { index = i; @@ -609,115 +529,102 @@ int dynType_complex_indexForName(dyn_type *type, const char *name) { return index; } -int dynType_complex_dynTypeAt(dyn_type *type, int index, dyn_type **result) { +const dyn_type* dynType_complex_dynTypeAt(const dyn_type* type, int index) { assert(type->type == DYN_TYPE_COMPLEX); assert(index >= 0); - dyn_type *sub = type->complex.types[index]; - if (sub->type == DYN_TYPE_REF) { - sub = sub->ref.ref; - } - *result = sub; - return 0; + return dynType_realType(type->complex.types[index]); } -int dynType_complex_setValueAt(dyn_type *type, int index, void *start, void *in) { +int dynType_complex_setValueAt(const dyn_type* type, int index, void* start, const void* in) { assert(type->type == DYN_TYPE_COMPLEX); - char *loc = ((char *)start) + dynType_getOffset(type, index); + char* loc = ((char*)start) + dynType_getOffset(type, index); size_t size = type->complex.structType.elements[index]->size; memcpy(loc, in, size); return 0; } -int dynType_complex_valLocAt(dyn_type *type, int index, void *inst, void **result) { +void* dynType_complex_valLocAt(const dyn_type* type, int index, void* inst) { assert(type->type == DYN_TYPE_COMPLEX); - char *l = (char *)inst; - void *loc = (void *)(l + dynType_getOffset(type, index)); - *result = loc; - return OK; + char* l = (char*)inst; + return (void* )(l + dynType_getOffset(type, index)); } -size_t dynType_complex_nrOfEntries(dyn_type *type) { +size_t dynType_complex_nrOfEntries(const dyn_type* type) { size_t count = 0; - struct complex_type_entry *entry = NULL; + struct complex_type_entry* entry = NULL; TAILQ_FOREACH(entry, &type->complex.entriesHead, entries) { ++count; } return count; } -int dynType_complex_entries(dyn_type *type, struct complex_type_entries_head **entries) { +const struct complex_type_entries_head* dynType_complex_entries(const dyn_type* type) { assert(type->type == DYN_TYPE_COMPLEX); - int status = OK; - *entries = &type->complex.entriesHead; - return status; + return &type->complex.entriesHead; } //sequence -void dynType_sequence_init(dyn_type *type, void *inst) { +void dynType_sequence_init(const dyn_type* type, void* inst) { assert(type->type == DYN_TYPE_SEQUENCE); - struct generic_sequence *seq = inst; + struct generic_sequence* seq = inst; seq->buf = NULL; seq->cap = 0; seq->len = 0; } -int dynType_sequence_alloc(dyn_type *type, void *inst, uint32_t cap) { +int dynType_sequence_alloc(const dyn_type* type, void* inst, uint32_t cap) { assert(type->type == DYN_TYPE_SEQUENCE); - int status = OK; - struct generic_sequence *seq = inst; - if (seq != NULL) { - size_t size = dynType_size(type->sequence.itemType); - seq->buf = calloc(cap, size); - if (seq->buf != NULL) { - seq->cap = cap; - seq->len = 0; - } else { - seq->cap = 0; - status = MEM_ERROR; - LOG_ERROR("Error allocating memory for buf"); - } - } else { - status = MEM_ERROR; - LOG_ERROR("Error allocating memory for seq"); - } - return status; + struct generic_sequence* seq = inst; + if (seq == NULL) { + celix_err_pushf("Error null sequence"); + return ERROR; + } + size_t size = dynType_size(type->sequence.itemType); + seq->buf = calloc(cap, size); + if (seq->buf == NULL) { + seq->cap = 0; + celix_err_pushf("Error allocating memory for seq buf"); + return MEM_ERROR; + } + seq->cap = cap; + seq->len = 0; + return OK; } -int dynType_sequence_reserve(dyn_type *type, void *inst, uint32_t cap) { +int dynType_sequence_reserve(const dyn_type* type, void* inst, uint32_t cap) { assert(type->type == DYN_TYPE_SEQUENCE); int status = OK; - struct generic_sequence *seq = inst; - if (seq != NULL && seq->cap < cap) { - size_t size = dynType_size(type->sequence.itemType); - seq->buf = realloc(seq->buf, (size_t)(cap * size)); - if (seq->buf != NULL) { - seq->cap = cap; - } else { - seq->cap = 0; - status = MEM_ERROR; - LOG_ERROR("Error allocating memory for buf"); - } - } else { - status = MEM_ERROR; - LOG_ERROR("Error allocating memory for seq"); - } + struct generic_sequence* seq = inst; + if (seq == NULL) { + celix_err_pushf("Error null sequence"); + return ERROR; + } + if (seq->cap >= cap) { + return OK; + } + size_t size = dynType_size(type->sequence.itemType); + seq->buf = realloc(seq->buf, (size_t)(cap * size)); + if (seq->buf == NULL) { + seq->cap = 0; + celix_err_pushf("Error allocating memory for seq buf"); + return MEM_ERROR; + } + memset(seq->buf+seq->cap*size, 0, (cap-seq->cap)*size); + seq->cap = cap; return status; } -void dynType_free(dyn_type *type, void *loc) { +void dynType_free(const dyn_type* type, void* loc) { dynType_deepFree(type, loc, true); } -void dynType_deepFree(dyn_type *type, void *loc, bool alsoDeleteSelf) { +static void dynType_deepFree(const dyn_type* type, void* loc, bool alsoDeleteSelf) { if (loc != NULL) { - dyn_type *subType = NULL; - char *text = NULL; + const dyn_type* subType = NULL; + char* text = NULL; + type = dynType_realType(type); switch (type->type) { - case DYN_TYPE_REF: - //NOTE: do not recursively forward asloDeleteSelf, because this is already handled in this function) - dynType_deepFree(type->ref.ref, loc, false); - break; case DYN_TYPE_COMPLEX : dynType_freeComplexType(type, loc); break; @@ -725,20 +632,22 @@ void dynType_deepFree(dyn_type *type, void *loc, bool alsoDeleteSelf) { dynType_freeSequenceType(type, loc); break; case DYN_TYPE_TYPED_POINTER: - dynType_typedPointer_getTypedType(type, &subType); + subType = dynType_typedPointer_getTypedType(type); void *ptrToType = *(void**)loc; dynType_deepFree(subType, ptrToType, true); break; case DYN_TYPE_TEXT : - text = *(char **)loc; + text = *(char**)loc; free(text); break; case DYN_TYPE_SIMPLE: //nop break; +//LCOV_EXCL_START default: - LOG_ERROR("Unexpected switch case. cannot free dyn type %c\n", type->descriptor); + assert(0 && "Unexpected switch case. cannot free dyn type"); break; +//LCOV_EXCL_STOP } if (alsoDeleteSelf) { @@ -747,11 +656,11 @@ void dynType_deepFree(dyn_type *type, void *loc, bool alsoDeleteSelf) { } } -void dynType_freeSequenceType(dyn_type *type, void *seqLoc) { - struct generic_sequence *seq = seqLoc; - dyn_type *itemType = dynType_sequence_itemType(type); - void *itemLoc = NULL; - int i; +static void dynType_freeSequenceType(const dyn_type* type, void* seqLoc) { + struct generic_sequence* seq = seqLoc; + const dyn_type* itemType = dynType_sequence_itemType(type); + void* itemLoc = NULL; + uint32_t i; for (i = 0; i < seq->len; ++i) { dynType_sequence_locForIndex(type, seqLoc, i, &itemLoc); dynType_deepFree(itemType, itemLoc, false); @@ -759,89 +668,72 @@ void dynType_freeSequenceType(dyn_type *type, void *seqLoc) { free(seq->buf); } -void dynType_freeComplexType(dyn_type *type, void *loc) { - struct complex_type_entry *entry = NULL; +static void dynType_freeComplexType(const dyn_type* type, void* loc) { + struct complex_type_entry* entry = NULL; int index = 0; - void *entryLoc = NULL; TAILQ_FOREACH(entry, &type->complex.entriesHead, entries) { - dynType_complex_valLocAt(type, index++, loc, &entryLoc); - dynType_deepFree(entry->type, entryLoc, false); + dynType_deepFree(entry->type, dynType_complex_valLocAt(type, index++, loc), false); } } -uint32_t dynType_sequence_length(void *seqLoc) { - struct generic_sequence *seq = seqLoc; +uint32_t dynType_sequence_length(const void *seqLoc) { + const struct generic_sequence* seq = seqLoc; return seq->len; } -int dynType_sequence_locForIndex(dyn_type *type, void *seqLoc, int index, void **out) { +int dynType_sequence_locForIndex(const dyn_type* type, const void* seqLoc, uint32_t index, void** out) { assert(type->type == DYN_TYPE_SEQUENCE); - int status = OK; - - struct generic_sequence *seq = seqLoc; + const struct generic_sequence* seq = seqLoc; size_t itemSize = dynType_size(type->sequence.itemType); if (index >= seq->cap) { - status = ERROR; - LOG_ERROR("Requested index (%i) is greater than capacity (%u) of sequence", index, seq->cap); + celix_err_pushf("Requested index (%u) is greater than capacity (%u) of sequence", index, seq->cap); + return ERROR; } if (index >= seq->len) { - status = ERROR; - LOG_ERROR("Requesting index (%i) outsize defined length (%u) but within capacity", index, seq->len); + celix_err_pushf("Requesting index (%u) outsize defined length (%u) but within capacity", index, seq->len); + return ERROR; } - if (status == OK) { - char *valLoc = seq->buf + (index * itemSize); - (*out) = valLoc; - } + char* valLoc = seq->buf + (index * itemSize); + (*out) = valLoc; - return status; + return OK; } -int dynType_sequence_increaseLengthAndReturnLastLoc(dyn_type *type, void *seqLoc, void **valLoc) { +int dynType_sequence_increaseLengthAndReturnLastLoc(const dyn_type* type, void* seqLoc, void** valLoc) { assert(type->type == DYN_TYPE_SEQUENCE); - int status = OK; - struct generic_sequence *seq = seqLoc; - - int lastIndex = seq->len; - if (seq->len < seq->cap) { - seq->len += 1; - } else { - status = ERROR; - LOG_ERROR("Cannot increase sequence length beyond capacity (%u)", seq->cap); - } + struct generic_sequence* seq = seqLoc; - if (status == OK) { - status = dynType_sequence_locForIndex(type, seqLoc, lastIndex, valLoc); + uint32_t lastIndex = seq->len; + if (seq->len >= seq->cap) { + celix_err_pushf("Cannot increase sequence length beyond capacity (%u)", seq->cap); + return ERROR; } - - return status; + seq->len += 1; + return dynType_sequence_locForIndex(type, seqLoc, lastIndex, valLoc); } -dyn_type * dynType_sequence_itemType(dyn_type *type) { +const dyn_type* dynType_sequence_itemType(const dyn_type* type) { assert(type->type == DYN_TYPE_SEQUENCE); - dyn_type *itemType = type->sequence.itemType; - if (itemType->type == DYN_TYPE_REF) { - itemType = itemType->ref.ref; - } - return itemType; + return dynType_realType(type->sequence.itemType); } -void dynType_simple_setValue(dyn_type *type, void *inst, void *in) { +void dynType_simple_setValue(const dyn_type *type, void *inst, const void *in) { size_t size = dynType_size(type); memcpy(inst, in, size); } -char dynType_descriptorType(dyn_type *type) { +char dynType_descriptorType(const dyn_type* type) { return type->descriptor; } -const char * dynType_getMetaInfo(dyn_type *type, const char *name) { - const char *result = NULL; - struct meta_entry *entry = NULL; +const char* dynType_getMetaInfo(const dyn_type* type, const char* name) { + const char* result = NULL; + struct meta_entry* entry = NULL; TAILQ_FOREACH(entry, &type->metaProperties, entries) { if (strcmp(entry->name, name) == 0) { result = entry->value; @@ -851,17 +743,16 @@ const char * dynType_getMetaInfo(dyn_type *type, const char *name) { return result; } -int dynType_metaEntries(dyn_type *type, struct meta_properties_head **entries) { - *entries = &type->metaProperties; - return OK; +const struct meta_properties_head* dynType_metaEntries(const dyn_type* type) { + return &type->metaProperties; } -const char * dynType_getName(dyn_type *type) { +const char* dynType_getName(const dyn_type* type) { return type->name; } -static ffi_type * dynType_ffiTypeFor(int c) { - ffi_type *type = NULL; +static ffi_type* dynType_ffiTypeFor(int c) { + ffi_type* type = NULL; switch (c) { case 'Z' : type = &ffi_type_uint8; @@ -909,11 +800,11 @@ static ffi_type * dynType_ffiTypeFor(int c) { return type; } -static unsigned short dynType_getOffset(dyn_type *type, int index) { +static unsigned short dynType_getOffset(const dyn_type* type, int index) { assert(type->type == DYN_TYPE_COMPLEX); unsigned short offset = 0; - ffi_type *ffiType = &type->complex.structType; + const ffi_type* ffiType = &type->complex.structType; int i; for (i = 0; i <= index && ffiType->elements[i] != NULL; i += 1) { size_t size = ffiType->elements[i]->size; @@ -930,52 +821,43 @@ static unsigned short dynType_getOffset(dyn_type *type, int index) { return offset; } -size_t dynType_size(dyn_type *type) { - dyn_type *rType = type; - if (type->type == DYN_TYPE_REF) { - rType = type->ref.ref; - } +size_t dynType_size(const dyn_type* type) { + const dyn_type* rType = dynType_realType(type); return rType->ffiType->size; } -int dynType_type(dyn_type *type) { +int dynType_type(const dyn_type* type) { return type->type; } - -int dynType_typedPointer_getTypedType(dyn_type *type, dyn_type **out) { - assert(type->type == DYN_TYPE_TYPED_POINTER); - int status = 0; - - dyn_type *typedType = type->typedPointer.typedType; - if (typedType->type == DYN_TYPE_REF) { - typedType = typedType->ref.ref; +const dyn_type* dynType_realType(const dyn_type* type) { + const dyn_type* real = type; + while (real->type == DYN_TYPE_REF) { + real= real->ref.ref; } + return real; - *out = typedType; - return status; +} + +const dyn_type* dynType_typedPointer_getTypedType(const dyn_type* type) { + assert(type->type == DYN_TYPE_TYPED_POINTER); + return dynType_realType(type->typedPointer.typedType); } -int dynType_text_allocAndInit(dyn_type *type, void *textLoc, const char *value) { +int dynType_text_allocAndInit(const dyn_type* type, void* textLoc, const char* value) { assert(type->type == DYN_TYPE_TEXT); - int status = 0; - const char *str = strdup(value); - char const **loc = textLoc; - if (str != NULL) { - *loc = str; - } else { - status = ERROR; - LOG_ERROR("Cannot allocate memory for string"); + const char* str = strdup(value); + char const** loc = textLoc; + if (str == NULL) { + celix_err_pushf("Cannot allocate memory for string"); + return MEM_ERROR; } - return status; + *loc = str; + return OK; } - - - - -void dynType_print(dyn_type *type, FILE *stream) { +void dynType_print(const dyn_type *type, FILE *stream) { if (type != NULL) { dynType_printTypes(type, stream); @@ -993,11 +875,9 @@ static void dynType_printDepth(int depth, FILE *stream) { } } -static void dynType_printAny(char *name, dyn_type *type, int depth, FILE *stream) { - dyn_type *toPrint = type; - if (toPrint->type == DYN_TYPE_REF) { - toPrint = toPrint->ref.ref; - } +static void dynType_printAny(const char* name, const dyn_type* type, int depth, FILE *stream) { + const dyn_type* toPrint = dynType_realType(type); + name = (name != NULL) ? name : "(unnamed)"; switch(toPrint->type) { case DYN_TYPE_COMPLEX : dynType_printComplex(name, toPrint, depth, stream); @@ -1011,18 +891,24 @@ static void dynType_printAny(char *name, dyn_type *type, int depth, FILE *stream case DYN_TYPE_TYPED_POINTER : dynType_printTypedPointer(name, toPrint, depth, stream); break; + case DYN_TYPE_TEXT: + dynType_printText(name, toPrint, depth, stream); + break; +//LCOV_EXCL_START default : - fprintf(stream, "TODO Unsupported type %d\n", toPrint->type); + assert(0 && "Unexpected switch case. cannot print dyn type"); break; +//LCOV_EXCL_STOP } } -static void dynType_printComplex(char *name, dyn_type *type, int depth, FILE *stream) { +static void dynType_printComplex(const char* name, const dyn_type* type, int depth, FILE *stream) { if (type->name == NULL) { dynType_printDepth(depth, stream); - fprintf(stream, "%s: complex type (anon), size is %zu, alignment is %i, descriptor is '%c'. fields:\n", name, type->ffiType->size, type->ffiType->alignment, type->descriptor); + fprintf(stream, "%s: complex type (anon), size is %zu, alignment is %i, descriptor is '%c'. fields:\n", + name, type->ffiType->size, type->ffiType->alignment, type->descriptor); - struct complex_type_entry *entry = NULL; + struct complex_type_entry* entry = NULL; TAILQ_FOREACH(entry, &type->complex.entriesHead, entries) { dynType_printAny(entry->name, entry->type, depth + 1, stream); } @@ -1031,49 +917,57 @@ static void dynType_printComplex(char *name, dyn_type *type, int depth, FILE *st fprintf(stream, "}\n"); } else { dynType_printDepth(depth, stream); - fprintf(stream, "%s: complex type ('%s'), size is %zu, alignment is %i, descriptor is '%c'.\n", name, type->name, type->ffiType->size, type->ffiType->alignment, type->descriptor); + fprintf(stream, "%s: complex type ('%s'), size is %zu, alignment is %i, descriptor is '%c'.\n", + name, type->name, type->ffiType->size, type->ffiType->alignment, type->descriptor); } } -static void dynType_printSequence(char *name, dyn_type *type, int depth, FILE *stream) { +static void dynType_printSequence(const char* name, const dyn_type* type, int depth, FILE* stream) { dynType_printDepth(depth, stream); - fprintf(stream, "sequence, size is %zu, alignment is %i, descriptor is '%c'. fields:\n", type->ffiType->size, type->ffiType->alignment, type->descriptor); + fprintf(stream, "sequence, size is %zu, alignment is %i, descriptor is '%c'. fields:\n", + type->ffiType->size, type->ffiType->alignment, type->descriptor); dynType_printDepth(depth + 1, stream); - fprintf(stream, "cap: simple type, size is %zu, alignment is %i.\n", type->sequence.seqType.elements[0]->size, type->sequence.seqType.elements[0]->alignment); + fprintf(stream, "cap: simple type, size is %zu, alignment is %i.\n", + type->sequence.seqType.elements[0]->size, type->sequence.seqType.elements[0]->alignment); dynType_printDepth(depth + 1, stream); - fprintf(stream, "len: simple type, size is %zu, alignment is %i.\n", type->sequence.seqType.elements[1]->size, type->sequence.seqType.elements[1]->alignment); + fprintf(stream, "len: simple type, size is %zu, alignment is %i.\n", + type->sequence.seqType.elements[1]->size, type->sequence.seqType.elements[1]->alignment); dynType_printDepth(depth + 1, stream); - fprintf(stream, "buf: array, size is %zu, alignment is %i. points to ->\n", type->sequence.seqType.elements[2]->size, type->sequence.seqType.elements[2]->alignment); + fprintf(stream, "buf: array, size is %zu, alignment is %i. points to ->\n", + type->sequence.seqType.elements[2]->size, type->sequence.seqType.elements[2]->alignment); dynType_printAny("element", type->sequence.itemType, depth + 1, stream); } -static void dynType_printSimple(char *name, dyn_type *type, int depth, FILE *stream) { +static void dynType_printSimple(const char* name, const dyn_type* type, int depth, FILE* stream) { if (type->descriptor != 'E') { dynType_printDepth(depth, stream); - fprintf(stream, "%s: simple type, size is %zu, alignment is %i, descriptor is '%c'.\n", name, type->ffiType->size, type->ffiType->alignment, type->descriptor); + fprintf(stream, "%s: simple type, size is %zu, alignment is %i, descriptor is '%c'.\n", + name, type->ffiType->size, type->ffiType->alignment, type->descriptor); } else { dynType_printEnum(name, type, depth, stream); } } -static void dynType_printEnum(char *name, dyn_type *type, int depth, FILE *stream) { +static void dynType_printEnum(const char* name, const dyn_type* type, int depth, FILE* stream) { dynType_printDepth(depth, stream); - fprintf(stream, "%s: enum type, size is %zu, alignment is %i, descriptor is '%c'. values:", name, type->ffiType->size, type->ffiType->alignment, type->descriptor); - struct meta_entry * m_entry; + fprintf(stream, "%s: enum type, size is %zu, alignment is %i, descriptor is '%c'. values:", + name, type->ffiType->size, type->ffiType->alignment, type->descriptor); + struct meta_entry* m_entry; TAILQ_FOREACH(m_entry, &type->metaProperties, entries) { fprintf(stream, " (\"%s\":\"%s\")", m_entry->name, m_entry->value); } fprintf(stream, "\n"); } -static void dynType_printTypedPointer(char *name, dyn_type *type, int depth, FILE *stream) { +static void dynType_printTypedPointer(const char* name, const dyn_type* type, int depth, FILE* stream) { dynType_printDepth(depth, stream); - fprintf(stream, "%s: typed pointer, size is %zu, alignment is %i, points to ->\n", name, type->ffiType->size, type->ffiType->alignment); - char *subName = NULL; + fprintf(stream, "%s: typed pointer, size is %zu, alignment is %i, points to ->\n", + name, type->ffiType->size, type->ffiType->alignment); + char* subName = NULL; char buf[128]; memset(buf,0,128); if (name != NULL) { @@ -1083,26 +977,30 @@ static void dynType_printTypedPointer(char *name, dyn_type *type, int depth, FIL dynType_printAny(subName, type->typedPointer.typedType, depth + 1, stream); } -static void dynType_printTypes(dyn_type *type, FILE *stream) { +static void dynType_printText(const char* name, const dyn_type* type, int depth, FILE* stream) { + dynType_printDepth(depth, stream); + fprintf(stream, "%s: text type, size is %zu, alignment is %i, descriptor is '%c'.\n", + name, type->ffiType->size, type->ffiType->alignment, type->descriptor); +} - dyn_type *parent = type->parent; - struct type_entry *pentry = NULL; - while (parent != NULL) { - TAILQ_FOREACH(pentry, &parent->nestedTypesHead, entries) { - if (pentry->type == type) { - return; +static void dynType_printTypes(const dyn_type* type, FILE* stream) { + if (type->type == DYN_TYPE_REF) { + const dyn_type* real = dynType_realType(type->ref.ref); + dyn_type* parent = type->parent; + struct type_entry* pentry = NULL; + while (parent != NULL) { + TAILQ_FOREACH(pentry, &parent->nestedTypesHead, entries) { + if (pentry->type == real) { + return; + } } + parent = parent->parent; } - parent = parent->parent; } - struct type_entry *entry = NULL; + struct type_entry* entry = NULL; TAILQ_FOREACH(entry, &type->nestedTypesHead, entries) { - dyn_type *toPrint = entry->type; - if (toPrint->type == DYN_TYPE_REF) { - toPrint = toPrint->ref.ref; - } - + const dyn_type* toPrint = dynType_realType(entry->type); switch(toPrint->type) { case DYN_TYPE_COMPLEX : dynType_printComplexType(toPrint, stream); @@ -1111,13 +1009,13 @@ static void dynType_printTypes(dyn_type *type, FILE *stream) { dynType_printSimpleType(toPrint, stream); break; default : - printf("TODO Print Type\n"); + printf("TODO Print Type %d\n", toPrint->type); break; } } - struct complex_type_entry *centry = NULL; + struct complex_type_entry* centry = NULL; switch(type->type) { case DYN_TYPE_COMPLEX : TAILQ_FOREACH(centry, &type->complex.entriesHead, entries) { @@ -1133,7 +1031,7 @@ static void dynType_printTypes(dyn_type *type, FILE *stream) { } } -static void dynType_printComplexType(dyn_type *type, FILE *stream) { +static void dynType_printComplexType(const dyn_type *type, FILE *stream) { fprintf(stream, "type '%s': complex type, size is %zu, alignment is %i, descriptor is '%c'. fields:\n", type->name, type->ffiType->size, type->ffiType->alignment, type->descriptor); struct complex_type_entry *entry = NULL; @@ -1144,7 +1042,11 @@ static void dynType_printComplexType(dyn_type *type, FILE *stream) { fprintf(stream, "}\n"); } -static void dynType_printSimpleType(dyn_type *type, FILE *stream) { - fprintf(stream, "\ttype '%s': simple type, size is %zu, alignment is %i, descriptor is '%c'\n", type->name, type->ffiType->size, type->ffiType->alignment, type->descriptor); +static void dynType_printSimpleType(const dyn_type *type, FILE *stream) { + fprintf(stream, "type '%s': simple type, size is %zu, alignment is %i, descriptor is '%c'\n", type->name, type->ffiType->size, type->ffiType->alignment, type->descriptor); +} + +bool dynType_isTrivial(const dyn_type* type) { + return dynType_realType(type)->trivial; } diff --git a/libs/dfi/src/dyn_type_common.c b/libs/dfi/src/dyn_type_common.c index 4db783c2d..96ea84d11 100644 --- a/libs/dfi/src/dyn_type_common.c +++ b/libs/dfi/src/dyn_type_common.c @@ -18,8 +18,8 @@ */ #include "dyn_type.h" - #include "dyn_type_common.h" +#include "celix_err.h" #include #include @@ -27,53 +27,31 @@ #include #include -DFI_SETUP_LOG(dynTypeCommon) - -dyn_type * dynType_findType(dyn_type *type, char *name) { - dyn_type *result = NULL; - - struct type_entry *entry = NULL; - if (type->referenceTypes != NULL) { - TAILQ_FOREACH(entry, type->referenceTypes, entries) { - if (strcmp(name, entry->type->name) == 0) { - result = entry->type; - break; +dyn_type* dynType_findType(dyn_type *type, char *name) { + for (dyn_type *current = type; current != NULL; current = current->parent) { + struct type_entry *entry = NULL; + if (current->name != NULL && strcmp(current->name, name) == 0) { + return current; + } + if (current->referenceTypes != NULL) { + TAILQ_FOREACH(entry, current->referenceTypes, entries) { + if (strcmp(name, entry->type->name) == 0) { + return entry->type; + } } } - } - if (result == NULL) { struct type_entry *nEntry = NULL; - TAILQ_FOREACH(nEntry, &type->nestedTypesHead, entries) { + TAILQ_FOREACH(nEntry, ¤t->nestedTypesHead, entries) { if (strcmp(name, nEntry->type->name) == 0) { - result = nEntry->type; - break; + return nEntry->type; } } } - - if (result == NULL && type->parent != NULL) { - result = dynType_findType(type->parent, name); - } - - return result; -} - -ffi_type * dynType_ffiType(dyn_type * type) { - if (type->type == DYN_TYPE_REF) { - if (type->ref.ref == NULL) { - LOG_ERROR("Error. Ref for %s is not (yet) initialized", type->name); - return NULL; - } - return type->ref.ref->ffiType; - } - return type->ffiType; + return NULL; } -void dynType_prepCif(ffi_type *type) { - ffi_cif cif; - ffi_type *args[1]; - args[0] = type; - ffi_prep_cif(&cif, FFI_DEFAULT_ABI, 1, &ffi_type_uint, args); +ffi_type* dynType_ffiType(const dyn_type * type) { + return dynType_realType(type)->ffiType; } diff --git a/libs/dfi/src/dyn_type_common.h b/libs/dfi/src/dyn_type_common.h index 856b9d321..aa5cc2aa4 100644 --- a/libs/dfi/src/dyn_type_common.h +++ b/libs/dfi/src/dyn_type_common.h @@ -24,46 +24,44 @@ #include "dyn_type.h" #include +#include -#include "dfi_log_util.h" #ifdef __cplusplus extern "C" { #endif -DFI_SETUP_LOG_HEADER(dynTypeCommon); - struct _dyn_type { - char *name; + char* name; char descriptor; + bool trivial; int type; - ffi_type *ffiType; - dyn_type *parent; - struct types_head *referenceTypes; //NOTE: not owned + ffi_type* ffiType; + dyn_type* parent; + const struct types_head* referenceTypes; //NOTE: not owned struct types_head nestedTypesHead; struct meta_properties_head metaProperties; union { struct { struct complex_type_entries_head entriesHead; ffi_type structType; //dyn_type.ffiType points to this - dyn_type **types; //based on entriesHead for fast access + dyn_type** types; //based on entriesHead for fast access } complex; struct { ffi_type seqType; //dyn_type.ffiType points to this - dyn_type *itemType; + dyn_type* itemType; } sequence; struct { - dyn_type *typedType; + dyn_type* typedType; } typedPointer; struct { - dyn_type *ref; + dyn_type* ref; } ref; }; }; -dyn_type * dynType_findType(dyn_type *type, char *name); -ffi_type * dynType_ffiType(dyn_type * type); -void dynType_prepCif(ffi_type *type); +dyn_type* dynType_findType(dyn_type* type, char* name); +ffi_type* dynType_ffiType(const dyn_type* type); #ifdef __cplusplus } diff --git a/libs/dfi/src/json_rpc.c b/libs/dfi/src/json_rpc.c index c2f7f43dc..d6547be9c 100644 --- a/libs/dfi/src/json_rpc.c +++ b/libs/dfi/src/json_rpc.c @@ -21,380 +21,300 @@ #include "json_serializer.h" #include "dyn_type.h" #include "dyn_interface.h" +#include "dyn_type_common.h" +#include "celix_cleanup.h" +#include "celix_err.h" + #include -#include #include #include #include -#include "celix_compiler.h" -#include "dyn_type_common.h" + +#define CELIX_JSON_RPC_MAX_ARGS 16 static int OK = 0; static int ERROR = 1; -DFI_SETUP_LOG(jsonRpc); - typedef void (*gen_func_type)(void); struct generic_service_layout { - void *handle; - gen_func_type methods[]; + void* handle; + gen_func_type methods[]; }; -int jsonRpc_call(dyn_interface_type *intf, void *service, const char *request, char **out) { - int status = OK; - - dyn_type* returnType = NULL; - - json_error_t error; - json_t *js_request = json_loads(request, 0, &error); - json_t *arguments = NULL; - const char *sig; - if (js_request) { - if (json_unpack(js_request, "{s:s}", "m", &sig) != 0) { - LOG_ERROR("Got json error '%s'\n", error.text); - json_decref(js_request); - return ERROR; - } else { - arguments = json_object_get(js_request, "a"); - } - } else { - LOG_ERROR("Got json error '%s' for '%s'\n", error.text, request); - return ERROR; - } - - struct methods_head *methods = NULL; - dynInterface_methods(intf, &methods); - struct method_entry *entry = NULL; - struct method_entry *method = NULL; - TAILQ_FOREACH(entry, methods, entries) { - if (strcmp(sig, entry->id) == 0) { - method = entry; - break; - } - } - - if (method == NULL) { - status = ERROR; - LOG_ERROR("Cannot find method with sig '%s'", sig); - } - else if (status == OK) { - returnType = dynFunction_returnType(method->dynFunc); - } - - void (*fp)(void) = NULL; - void *handle = NULL; - if (status == OK) { - struct generic_service_layout *serv = service; - handle = serv->handle; - fp = serv->methods[method->index]; - } - - dyn_function_type *func = NULL; - int nrOfArgs = 0; - if (status == OK) { - nrOfArgs = dynFunction_nrOfArguments(entry->dynFunc); - func = entry->dynFunc; - } - - void *args[nrOfArgs]; - - json_t *value = NULL; - - int i; - int index = 0; - - void *ptr = NULL; - void *ptrToPtr = &ptr; - - //setup and deserialize input - for (i = 0; i < nrOfArgs; ++i) { - dyn_type *argType = dynFunction_argumentTypeForIndex(func, i); - enum dyn_function_argument_meta meta = dynFunction_argumentMetaForIndex(func, i); - if (meta == DYN_FUNCTION_ARGUMENT_META__STD) { - value = json_array_get(arguments, index++); - void *outPtr = NULL; - status = jsonSerializer_deserializeJson(argType, value, &outPtr); - args[i] = outPtr; - } else if (meta == DYN_FUNCTION_ARGUMENT_META__PRE_ALLOCATED_OUTPUT) { - void **instPtr = calloc(1, sizeof(void*)); - void *inst = NULL; - dyn_type *subType = NULL; - dynType_typedPointer_getTypedType(argType, &subType); - dynType_alloc(subType, &inst); - *instPtr = inst; - args[i] = instPtr; - } else if (meta == DYN_FUNCTION_ARGUMENT_META__OUTPUT) { - args[i] = &ptrToPtr; - } else if (meta == DYN_FUNCTION_ARGUMENT_META__HANDLE) { - args[i] = &handle; - } - - if (status != OK) { - break; - } - } - json_decref(js_request); - - if (status == OK) { - if (dynType_descriptorType(returnType) != 'N') { - //NOTE To be able to handle exception only N as returnType is supported - LOG_ERROR("Only interface methods with a native int are supported. Found type '%c'", (char)dynType_descriptorType(returnType)); - status = ERROR; - } - } +typedef struct celix_rpc_args { + const struct dyn_function_arguments_head* dynArgs; + void* args[CELIX_JSON_RPC_MAX_ARGS]; +}celix_rpc_args_t; - ffi_sarg returnVal = 1; - - if (status == OK) { - status = dynFunction_call(func, fp, (void *) &returnVal, args); - } - - int funcCallStatus = (int)returnVal; - - //free input args - json_t *jsonResult = NULL; - for(i = 0; i < nrOfArgs; ++i) { - dyn_type *argType = dynFunction_argumentTypeForIndex(func, i); - enum dyn_function_argument_meta meta = dynFunction_argumentMetaForIndex(func, i); - if (meta == DYN_FUNCTION_ARGUMENT_META__STD) { - if (dynType_descriptorType(argType) == 't') { - const char* isConst = dynType_getMetaInfo(argType, "const"); - if (isConst != NULL && strncmp("true", isConst, 5) == 0) { - dynType_free(argType, args[i]); - } else { +static void celix_rpcArgs_cleanup(celix_rpc_args_t* args) { + const struct dyn_function_arguments_head* dynArgs = args->dynArgs; + if (dynArgs == NULL) { + return; + } + dyn_function_argument_type* entry = NULL; + TAILQ_FOREACH(entry, dynArgs, entries) { + const dyn_type* argType = dynType_realType(entry->type); + enum dyn_function_argument_meta meta = entry->argumentMeta; + if (meta == DYN_FUNCTION_ARGUMENT_META__STD) { + if (dynType_descriptorType(argType) == 't') { + const char* isConst = dynType_getMetaInfo(entry->type, "const"); + if (isConst != NULL && strncmp("true", isConst, 5) == 0) { + dynType_free(argType, args->args[entry->index]); + } else { //char* -> callee is now owner, no free for char seq needed //will free the actual pointer - free(args[i]); - } - } else { - dynType_free(argType, args[i]); + free(args->args[entry->index]); + } + } else { + dynType_free(argType, args->args[entry->index]); } - } - } - - //serialize and free output - for (i = 0; i < nrOfArgs; i += 1) { - dyn_type *argType = dynFunction_argumentTypeForIndex(func, i); - enum dyn_function_argument_meta meta = dynFunction_argumentMetaForIndex(func, i); - if (meta == DYN_FUNCTION_ARGUMENT_META__PRE_ALLOCATED_OUTPUT) { - if (funcCallStatus == 0 && status == OK) { - status = jsonSerializer_serializeJson(argType, args[i], &jsonResult); - } - dyn_type *subType = NULL; - dynType_typedPointer_getTypedType(argType, &subType); - void **ptrToInst = (void**)args[i]; - dynType_free(subType, *ptrToInst); - free(ptrToInst); - } else if (meta == DYN_FUNCTION_ARGUMENT_META__OUTPUT) { - if (funcCallStatus == 0 && ptr != NULL) { - dyn_type *typedType = NULL; - if (status == OK) { - status = dynType_typedPointer_getTypedType(argType, &typedType); - } - if (status == OK && dynType_descriptorType(typedType) == 't') { - status = jsonSerializer_serializeJson(typedType, (void*) &ptr, &jsonResult); - free(ptr); - } else { - dyn_type *typedTypedType = NULL; - if (status == OK) { - status = dynType_typedPointer_getTypedType(typedType, &typedTypedType); - } - - if(status == OK){ - status = jsonSerializer_serializeJson(typedTypedType, ptr, &jsonResult); - } + } else if (meta == DYN_FUNCTION_ARGUMENT_META__PRE_ALLOCATED_OUTPUT) { + const dyn_type* subType = dynType_typedPointer_getTypedType(argType); + dynType_free(subType, *(void**)(args->args[entry->index])); + } else if (meta == DYN_FUNCTION_ARGUMENT_META__OUTPUT) { + const dyn_type* typedType = dynType_typedPointer_getTypedType(argType); + if (dynType_descriptorType(typedType) == 't') { + free(**(void***)args->args[entry->index]); + } else { + const dyn_type* typedTypedType = dynType_typedPointer_getTypedType(typedType); + dynType_free(typedTypedType, **(void***)args->args[entry->index]); + } + } + } + args->dynArgs = NULL; +} - if (status == OK) { - dynType_free(typedTypedType, ptr); - } - } +CELIX_DEFINE_AUTO_CLEANUP_CLEAR_FUNC(celix_rpc_args_t, celix_rpcArgs_cleanup) - } - } +int jsonRpc_call(const dyn_interface_type* intf, void* service, const char* request, char** out) { + int status = OK; - if (status != OK) { - break; - } - } + json_error_t error; + json_auto_t* js_request = json_loads(request, 0, &error); + if (js_request == NULL) { + celix_err_pushf("Got json error: %s", error.text); + return ERROR; + } + json_t* arguments = NULL; + const char* sig; + if (json_unpack(js_request, "{s:s}", "m", &sig) != 0) { + celix_err_push("Error getting method signature"); + return ERROR; + } + arguments = json_object_get(js_request, "a"); + if (arguments == NULL || !json_is_array(arguments)) { + celix_err_pushf("Error getting arguments array for %s", sig); + return ERROR; + } - char *response = NULL; - if (status == OK) { - json_t *payload = json_object(); - if (funcCallStatus == 0) { - if (jsonResult == NULL) { - //ignore -> no result - } else { - json_object_set_new_nocheck(payload, "r", jsonResult); - } - } else { - json_object_set_new_nocheck(payload, "e", json_integer(funcCallStatus)); - } - response = json_dumps(payload, JSON_COMPACT | JSON_ENCODE_ANY);//Should use JSON_COMPACT, it can reduce the size of the JSON string. - json_decref(payload); - } + const struct method_entry* method = dynInterface_findMethod(intf, sig); + if (method == NULL) { + celix_err_pushf("Cannot find method with sig '%s'", sig); + return ERROR; + } - if (status == OK) { - *out = response; - } else { - free(response); - } + struct generic_service_layout* serv = service; + const struct dyn_function_arguments_head* dynArgs = dynFunction_arguments(method->dynFunc); + const dyn_function_argument_type* last = TAILQ_LAST(dynArgs, dyn_function_arguments_head); + int nrOfArgs = dynFunction_nrOfArguments(method->dynFunc); + if (nrOfArgs > CELIX_JSON_RPC_MAX_ARGS) { + celix_err_pushf("Too many arguments for %s: %d > %d", sig, nrOfArgs, CELIX_JSON_RPC_MAX_ARGS); + return ERROR; + } + void* ptr = NULL; + void* ptrToPtr = &ptr; + celix_auto(celix_rpc_args_t) rpcArgs = { dynArgs, {0} }; + + rpcArgs.args[0] = &serv->handle; + --nrOfArgs; + if (last->argumentMeta == DYN_FUNCTION_ARGUMENT_META__PRE_ALLOCATED_OUTPUT) { + const dyn_type *subType = dynType_typedPointer_getTypedType(dynType_realType(last->type)); + rpcArgs.args[last->index] = &ptr; + if ((status = dynType_alloc(subType, &ptr)) != OK) { + celix_err_pushf("Error allocating memory for pre-allocated output argument of %s", sig); + return ERROR; + } + --nrOfArgs; + } else if (last->argumentMeta == DYN_FUNCTION_ARGUMENT_META__OUTPUT) { + rpcArgs.args[last->index] = &ptrToPtr; + --nrOfArgs; + } + if ((size_t)nrOfArgs != json_array_size(arguments)) { + celix_err_pushf("Wrong number of standard arguments for %s. Expected %d, got %zu", + sig, nrOfArgs, json_array_size(arguments)); + return ERROR; + } + //setup and deserialize input + dyn_function_argument_type* entry = NULL; + TAILQ_FOREACH(entry, dynArgs, entries) { + if (entry->argumentMeta != DYN_FUNCTION_ARGUMENT_META__STD) { + continue; + } + status = jsonSerializer_deserializeJson(entry->type, json_array_get(arguments, entry->index-1), &(rpcArgs.args[entry->index])); + if (status != OK) { + celix_err_pushf("Error deserializing argument %d for %s", entry->index, sig); + return status; + } + } + ffi_sarg returnVal = 1; + (void)dynFunction_call(method->dynFunc, serv->methods[method->index], (void *) &returnVal, rpcArgs.args); + + int funcCallStatus = (int)returnVal; + //serialize output + json_auto_t* jsonResult = NULL; + if (funcCallStatus == 0) { + const dyn_type* argType = dynType_realType(last->type); + if (last->argumentMeta == DYN_FUNCTION_ARGUMENT_META__PRE_ALLOCATED_OUTPUT) { + status = jsonSerializer_serializeJson(argType, rpcArgs.args[last->index], &jsonResult); + } else if (last->argumentMeta == DYN_FUNCTION_ARGUMENT_META__OUTPUT) { + status = jsonSerializer_serializeJson(dynType_typedPointer_getTypedType(argType), (void*) &ptr, &jsonResult); + } + if (status != OK) { + celix_err_pushf("Error serializing result for %s", sig); + return status; + } + } + celix_rpcArgs_cleanup(&rpcArgs); - return status; + json_auto_t* payload = json_object(); + if (funcCallStatus == 0) { + if (jsonResult != NULL) { + status = json_object_set_new_nocheck(payload, "r", celix_steal_ptr(jsonResult)); + } + } else { + status = json_object_set_new_nocheck(payload, "e", json_integer(funcCallStatus)); + } + if (status != 0) { + celix_err_pushf("Error generating response payload for %s", sig); + return ERROR; + } + //use JSON_COMPACT to reduce the size of the JSON string. + *out = json_dumps(payload, JSON_COMPACT | JSON_ENCODE_ANY); + return (*out != NULL) ? OK : ERROR; } -int jsonRpc_prepareInvokeRequest(dyn_function_type *func, const char *id, void *args[], char **out) { - int status = OK; - - - json_t *invoke = json_object(); - json_object_set_new_nocheck(invoke, "m", json_string(id)); - - json_t *arguments = json_array(); - json_object_set_new_nocheck(invoke, "a", arguments); +int jsonRpc_prepareInvokeRequest(const dyn_function_type* func, const char* id, void* args[], char** out) { + json_auto_t* invoke = json_object(); + // each method must have a non-null id + if (json_object_set_new_nocheck(invoke, "m", json_string(id)) != 0) { + celix_err_pushf("Error setting method name '%s'", id); + return ERROR; + } - int i; - int nrOfArgs = dynFunction_nrOfArguments(func); - for (i = 0; i < nrOfArgs; i +=1) { - dyn_type *type = dynFunction_argumentTypeForIndex(func, i); - enum dyn_function_argument_meta meta = dynFunction_argumentMetaForIndex(func, i); - if (meta == DYN_FUNCTION_ARGUMENT_META__STD) { - json_t *val = NULL; + json_auto_t* arguments = json_array(); + if (json_object_set_nocheck(invoke, "a", arguments) != 0) { + celix_err_pushf("Error adding arguments array for '%s'", id); + return ERROR; + } - int rc = jsonSerializer_serializeJson(type, args[i], &val); + const struct dyn_function_arguments_head* dynArgs = dynFunction_arguments(func); + dyn_function_argument_type* entry = NULL; + TAILQ_FOREACH(entry, dynArgs, entries) { + const dyn_type* type = dynType_realType(entry->type); + enum dyn_function_argument_meta meta = entry->argumentMeta; + if (meta == DYN_FUNCTION_ARGUMENT_META__STD) { + json_t* val = NULL; + + int rc = jsonSerializer_serializeJson(type, args[entry->index], &val); + if (rc != 0) { + celix_err_pushf("Failed to serialize args for function '%s'\n", id); + return ERROR; + } if (dynType_descriptorType(type) == 't') { - const char *metaArgument = dynType_getMetaInfo(type, "const"); - if (metaArgument != NULL && strncmp("true", metaArgument, 5) == 0) { + // we need to get meta info from the original type, which could be a reference, rather than the real type + const char* metaArgument = dynType_getMetaInfo(entry->type, "const"); + if (metaArgument != NULL && strcmp("true", metaArgument) == 0) { //const char * as input -> nop } else { - char **str = args[i]; + char** str = args[entry->index]; free(*str); //char * as input -> got ownership -> free it. } } - - if (rc == 0) { - json_array_append_new(arguments, val); - } else { - LOG_ERROR("Failed to serialize args for function '%s'\n", id); - status = ERROR; - break; - } - } else { - //skip handle / output types - } - } - - char *invokeStr = json_dumps(invoke, JSON_COMPACT | JSON_ENCODE_ANY);//Should use JSON_COMPACT, it can reduce the size of the JSON string. - json_decref(invoke); - - if (status == OK) { - *out = invokeStr; - } else { - *out = NULL; - free(invokeStr); + if (json_array_append_new(arguments, val) != 0) { + celix_err_pushf("Error adding argument (%d) for '%s'", entry->index, id); + return ERROR; + } + } } - return status; + //use JSON_COMPACT to reduce the size of the JSON string. + char* invokeStr = json_dumps(invoke, JSON_COMPACT | JSON_ENCODE_ANY); + *out = invokeStr; + return *out != NULL ? OK : ERROR; } -int jsonRpc_handleReply(dyn_function_type *func, const char *reply, void *args[], int *rsErrno) { - int status = OK; +int jsonRpc_handleReply(const dyn_function_type* func, const char* reply, void* args[], int* rsErrno) { + int status = OK; - json_error_t error; - json_t *replyJson = json_loads(reply, JSON_DECODE_ANY, &error); - if (replyJson == NULL) { - status = ERROR; - LOG_ERROR("Error parsing json '%s', got error '%s'", reply, error.text); - } - - json_t *result = NULL; - json_t *rsError = NULL; - bool replyHasError = false; - if (status == OK) { - *rsErrno = 0; - result = json_object_get(replyJson, "r"); - if (result == NULL) { - rsError = json_object_get(replyJson, "e"); - if(rsError != NULL) { - //get the invocation error of remote service function - *rsErrno = (int)json_integer_value(rsError); - replyHasError = true; - } - } - } + json_error_t error; + json_auto_t* replyJson = json_loads(reply, JSON_DECODE_ANY, &error); + if (replyJson == NULL) { + celix_err_pushf("Error parsing json '%s', got error '%s'", reply, error.text); + return ERROR; + } - int nrOfOutputArgs = 0; - int nrOfArgs = dynFunction_nrOfArguments(func); - for (int j = 0; j < nrOfArgs; ++j) { - enum dyn_function_argument_meta meta = dynFunction_argumentMetaForIndex(func, j); - if (meta == DYN_FUNCTION_ARGUMENT_META__PRE_ALLOCATED_OUTPUT || meta == DYN_FUNCTION_ARGUMENT_META__OUTPUT) { - nrOfOutputArgs += 1; - if (nrOfOutputArgs > 1) { - status = ERROR; - LOG_ERROR("Only one output argument is supported"); - break; + json_t* result = NULL; + json_t* rsError = NULL; + *rsErrno = 0; + + const struct dyn_function_arguments_head* arguments = dynFunction_arguments(func); + dyn_function_argument_type* last = TAILQ_LAST(arguments, dyn_function_arguments_head); + const dyn_type* argType = dynType_realType(last->type); + enum dyn_function_argument_meta meta = last->argumentMeta; + rsError = json_object_get(replyJson, "e"); + if (rsError != NULL) { + //get the invocation error of remote service function + *rsErrno = (int) json_integer_value(rsError); + return OK; + } + if (meta != DYN_FUNCTION_ARGUMENT_META__PRE_ALLOCATED_OUTPUT && meta != DYN_FUNCTION_ARGUMENT_META__OUTPUT) { + return OK; + } + result = json_object_get(replyJson, "r"); + if (result == NULL) { + celix_err_pushf("Expected result in reply. got '%s'", reply); + return ERROR; + } + void** lastArg = (void **) args[last->index]; + if (*lastArg == NULL) { + // caller provides nullptr, no need to deserialize + return OK; + } + if (meta == DYN_FUNCTION_ARGUMENT_META__PRE_ALLOCATED_OUTPUT) { + void* tmp = NULL; + void** out = lastArg; + size_t size = 0; + + const dyn_type* subType = dynType_typedPointer_getTypedType(argType); + status = jsonSerializer_deserializeJson(subType, result, &tmp); + if (tmp != NULL) { + size = dynType_size(subType); + memcpy(*out, tmp, size); + dynType_free(subType, tmp); + } + } else { + const dyn_type* subType = dynType_typedPointer_getTypedType(argType); + + if (dynType_descriptorType(subType) == 't') { + char*** out = (char ***) lastArg; + char** ptrToString = NULL; + status = jsonSerializer_deserializeJson(subType, result, (void**)&ptrToString); + if (ptrToString != NULL) { + **out = (void*)*ptrToString; + free(ptrToString); } - if (result == NULL && !replyHasError) { - status = ERROR; - LOG_ERROR("Expected result in reply. got '%s'", reply); - break; + } else { + const dyn_type* subSubType = dynType_typedPointer_getTypedType(subType); + void*** out = (void ***) lastArg; + if (json_is_null(result)) { + **out = NULL; + } else { + status = jsonSerializer_deserializeJson(subSubType, result, *out); } } } - if (status == OK && !replyHasError) { - int i; - for (i = 0; i < nrOfArgs; i += 1) { - dyn_type *argType = dynFunction_argumentTypeForIndex(func, i); - enum dyn_function_argument_meta meta = dynFunction_argumentMetaForIndex(func, i); - if (meta == DYN_FUNCTION_ARGUMENT_META__PRE_ALLOCATED_OUTPUT) { - void *tmp = NULL; - void **out = (void **) args[i]; - size_t size = 0; - - if (dynType_descriptorType(argType) == 't') { - status = jsonSerializer_deserializeJson(argType, result, &tmp); - if (tmp != NULL) { - size = strnlen(((char *) *(char**) tmp), 1024 * 1024); - memcpy(*out, *(void**) tmp, size); - } - } else { - dynType_typedPointer_getTypedType(argType, &argType); - status = jsonSerializer_deserializeJson(argType, result, &tmp); - if (tmp != NULL) { - size = dynType_size(argType); - memcpy(*out, tmp, size); - } - } - - dynType_free(argType, tmp); - } else if (meta == DYN_FUNCTION_ARGUMENT_META__OUTPUT) { - dyn_type *subType = NULL; - - dynType_typedPointer_getTypedType(argType, &subType); - - if (dynType_descriptorType(subType) == 't') { - char ***out = (char ***) args[i]; - char **ptrToString = NULL; - status = jsonSerializer_deserializeJson(subType, result, (void**)&ptrToString); - char *s CELIX_UNUSED = *ptrToString; //note for debug - free(ptrToString); - **out = (void*)s; - } else { - dyn_type *subSubType = NULL; - dynType_typedPointer_getTypedType(subType, &subSubType); - void ***out = (void ***) args[i]; - status = jsonSerializer_deserializeJson(subSubType, result, *out); - } - } else { - //skip - } - } - } - - - json_decref(replyJson); - - return status; + return status; } diff --git a/libs/dfi/src/json_serializer.c b/libs/dfi/src/json_serializer.c index e24cc58d4..ee8fa7a42 100644 --- a/libs/dfi/src/json_serializer.c +++ b/libs/dfi/src/json_serializer.c @@ -20,226 +20,158 @@ #include "json_serializer.h" #include "dyn_type.h" #include "dyn_type_common.h" -#include "dyn_interface.h" +#include "celix_err.h" #include #include #include #include -static int jsonSerializer_createType(dyn_type *type, json_t *object, void **result); -static int jsonSerializer_parseObject(dyn_type *type, json_t *object, void *inst); -static int jsonSerializer_parseObjectMember(dyn_type *type, const char *name, json_t *val, void *inst); -static int jsonSerializer_parseSequence(dyn_type *seq, json_t *array, void *seqLoc); -static int jsonSerializer_parseAny(dyn_type *type, void *input, json_t *val); -static int jsonSerializer_parseEnum(dyn_type *type, const char* enum_name, int32_t *out); +static int jsonSerializer_createType(const dyn_type* type, json_t* object, void** result); +static int jsonSerializer_parseObject(const dyn_type* type, json_t* object, void* inst); +static int jsonSerializer_parseSequence(const dyn_type* seq, json_t* array, void* seqLoc); +static int jsonSerializer_parseAny(const dyn_type* type, void* input, json_t* val); +static int jsonSerializer_parseEnum(const dyn_type* type, const char* enum_name, int32_t* out); -static int jsonSerializer_writeAny(dyn_type *type, void *input, json_t **val); -static int jsonSerializer_writeComplex(dyn_type *type, void *input, json_t **val); -static int jsonSerializer_writeSequence(dyn_type *type, void *input, json_t **out); -static int jsonSerializer_writeEnum(dyn_type *type, int32_t enum_value, json_t **out); +static int jsonSerializer_writeAny(const dyn_type* type, const void* input, json_t** val); +static int jsonSerializer_writeComplex(const dyn_type* type, const void* input, json_t** val); +static int jsonSerializer_writeSequence(const dyn_type* type, const void* input, json_t** out); +static int jsonSerializer_writeEnum(const dyn_type* type, int32_t enum_value, json_t** out); static int OK = 0; static int ERROR = 1; -DFI_SETUP_LOG(jsonSerializer); - -int jsonSerializer_deserialize(dyn_type *type, const char *input, size_t length, void **result) { - assert(dynType_type(type) == DYN_TYPE_COMPLEX || dynType_type(type) == DYN_TYPE_SEQUENCE); +int jsonSerializer_deserialize(const dyn_type* type, const char* input, size_t length, void** result) { int status = 0; json_error_t error; - json_t *root = json_loadb(input, length, JSON_DECODE_ANY, &error); - - if (root != NULL) { - status = jsonSerializer_deserializeJson(type, root, result); - json_decref(root); - } else { - status = ERROR; - LOG_ERROR("Error parsing json input '%.*s'. Error is: %s\n", (int)length, input, error.text); + json_auto_t* root = json_loadb(input, length, JSON_DECODE_ANY, &error); + if (root == NULL) { + celix_err_pushf("Error parsing json input '%.*s'. Error is: %s\n", (int)length, input, error.text); + return ERROR; } + status = jsonSerializer_deserializeJson(type, root, result); if (status != OK) { - LOG_ERROR("Error cannot deserialize json. Input is '%s'\n", input); + celix_err_pushf("Error cannot deserialize json. Input is '%s'", input); } return status; } -int jsonSerializer_deserializeJson(dyn_type *type, json_t *input, void **out) { - return jsonSerializer_createType(type, input, out); +int jsonSerializer_deserializeJson(const dyn_type* type, json_t* input, void** out) { + return jsonSerializer_createType(dynType_realType(type), input, out); } -static int jsonSerializer_createType(dyn_type *type, json_t *val, void **result) { +static int jsonSerializer_createType(const dyn_type* type, json_t* val, void** result) { assert(val != NULL); int status = OK; - void *inst = NULL; - - if (dynType_descriptorType(type) == 't') { - if (json_typeof(val) == JSON_STRING) { - //note a deserialized C string is a sequence of memory for the actual string and a - //pointer to that sequence. That pointer also needs to reside in the memory (heap). - const char *s = json_string_value(val); - inst = calloc(1, sizeof(char*)); - *((char**)inst) = strdup(s); - } else { - status = ERROR; - LOG_ERROR("Expected json_string type got %i\n", json_typeof(val)); - } - } else { - status = dynType_alloc(type, &inst); + void* inst = NULL; - if (status == OK) { - assert(inst != NULL); - status = jsonSerializer_parseAny(type, inst, val); - } + if ((status = dynType_alloc(type, &inst)) != OK) { + return status; } - if (status == OK) { - *result = inst; - } else { + if ((status = jsonSerializer_parseAny(type, inst, val)) != OK) { + dynType_free(type, inst); *result = NULL; - dynType_free(type, inst); + return status; } - return status; + *result = inst; + return OK; } -static int jsonSerializer_parseObject(dyn_type *type, json_t *object, void *inst) { +static int jsonSerializer_parseObject(const dyn_type* type, json_t* object, void* inst) { assert(object != NULL); - int status = 0; - json_t *value; - const char *key; - - json_object_foreach(object, key, value) { - status = jsonSerializer_parseObjectMember(type, key, value, inst); + int status = OK; + json_t* value; + const struct complex_type_entries_head* entries = dynType_complex_entries(type); + struct complex_type_entry* entry = NULL; + int index = 0; + void* valp = NULL; + const dyn_type* valType = NULL; + TAILQ_FOREACH(entry, entries, entries) { + if (entry->name == NULL) { + celix_err_push("Unamed field unsupported"); + return ERROR; + } + value = json_object_get(object, entry->name); + if (value == NULL) { + celix_err_pushf("Missing object member %s", entry->name); + return ERROR; + } + valp = dynType_complex_valLocAt(type, index, inst); + valType = dynType_complex_dynTypeAt(type, index); + status = jsonSerializer_parseAny(valType, valp, value); if (status != OK) { break; } + index++; } return status; } -static int jsonSerializer_parseObjectMember(dyn_type *type, const char *name, json_t *val, void *inst) { - int status = OK; - void *valp = NULL; - dyn_type *valType = NULL; - - int index = dynType_complex_indexForName(type, name); - if (index < 0) { - return OK;//We should ignore unknown name in request or response. Satisfy forward compatibility for responses. - } - - if (status == OK) { - status = dynType_complex_valLocAt(type, index, inst, &valp); - } - - if (status == OK ) { - status = dynType_complex_dynTypeAt(type, index, &valType); - } - - if (status == OK) { - status = jsonSerializer_parseAny(valType, valp, val); - } - - return status; -} - -static int jsonSerializer_parseAny(dyn_type *type, void *loc, json_t *val) { +static int jsonSerializer_parseAny(const dyn_type* type, void* loc, json_t* val) { int status = OK; - dyn_type *subType = NULL; + const dyn_type* subType = NULL; char c = dynType_descriptorType(type); - /* - printf("parseAny with descriptor '%c' :", c); - json_dumpf(val, stdout, 0); //TODO remove - printf("\n"); - */ - - bool *z; //Z - float *f; //F - double *d; //D - char *b; //B - int *n; //N - int16_t *s; //S - int32_t *i; //I - int32_t *E; //E - int64_t *l; //J - uint8_t *ub; //b - uint16_t *us; //s - uint32_t *ui; //i - uint64_t *ul; //j - switch (c) { case 'Z' : - z = loc; - *z = (bool) json_is_true(val); + *(bool*)loc = (bool) json_is_true(val); break; case 'F' : - f = loc; - *f = (float) json_real_value(val); + *(float*)loc = (float) json_real_value(val); break; case 'D' : - d = loc; - *d = json_real_value(val); + *(double*)loc = json_real_value(val); break; case 'N' : - n = loc; - *n = (int) json_integer_value(val); + *(int*)loc = (int) json_integer_value(val); break; case 'B' : - b = loc; - *b = (char) json_integer_value(val); + *(char*)loc = (char) json_integer_value(val); break; case 'S' : - s = loc; - *s = (int16_t) json_integer_value(val); + *(int16_t*)loc = (int16_t) json_integer_value(val); break; case 'I' : - i = loc; - *i = (int32_t) json_integer_value(val); + *(int32_t*)loc = (int32_t) json_integer_value(val); break; case 'J' : - l = loc; - *l = (int64_t) json_integer_value(val); + *(int64_t*)loc = (int64_t) json_integer_value(val); break; case 'b' : - ub = loc; - *ub = (uint8_t) json_integer_value(val); + *(uint8_t*)loc = (uint8_t) json_integer_value(val); break; case 's' : - us = loc; - *us = (uint16_t) json_integer_value(val); + *(uint16_t*)loc = (uint16_t) json_integer_value(val); break; case 'i' : - ui = loc; - *ui = (uint32_t) json_integer_value(val); + *(uint32_t*)loc = (uint32_t) json_integer_value(val); break; case 'j' : - ul = loc; - *ul = (uint64_t) json_integer_value(val); + *(uint64_t*)loc = (uint64_t) json_integer_value(val); break; case 'E' : - if (json_is_null(val)) { - //nop - } else if (json_is_string(val)){ - E = loc; - status = jsonSerializer_parseEnum(type, json_string_value(val), E); + if (json_is_string(val)){ + status = jsonSerializer_parseEnum(type, json_string_value(val), loc); } else { status = ERROR; - LOG_ERROR("Expected json string for enum type but got %i", json_typeof(val)); + celix_err_pushf("Expected json string for enum type but got %i", json_typeof(val)); } break; case 't' : if (json_is_null(val)) { - //nop + // NULL string is allowed } else if (json_is_string(val)) { - dynType_text_allocAndInit(type, loc, json_string_value(val)); + status = dynType_text_allocAndInit(type, loc, json_string_value(val)); } else { status = ERROR; - LOG_ERROR("Expected json string type got %i", json_typeof(val)); + celix_err_pushf("Expected json string type got %i", json_typeof(val)); } break; case '[' : @@ -247,78 +179,79 @@ static int jsonSerializer_parseAny(dyn_type *type, void *loc, json_t *val) { status = jsonSerializer_parseSequence(type, val, loc); } else { status = ERROR; - LOG_ERROR("Expected json array type got '%i'", json_typeof(val)); + celix_err_pushf("Expected json array type got '%i'", json_typeof(val)); } break; case '{' : - if (status == OK) { + if (json_is_object(val)) { status = jsonSerializer_parseObject(type, val, loc); + } else { + status = ERROR; + celix_err_pushf("Expected json object type got '%i'", json_typeof(val)); } break; case '*' : - status = dynType_typedPointer_getTypedType(type, &subType); - if (status == OK) { - status = jsonSerializer_createType(subType, val, (void **) loc); + subType = dynType_typedPointer_getTypedType(type); + if (dynType_ffiType(subType) != &ffi_type_pointer) { + // NULL pointer is allowed + if (!json_is_null(val)) { + status = jsonSerializer_createType(subType, val, (void **) loc); + } + } else { + status = ERROR; + celix_err_pushf("Error cannot deserialize pointer to pointer"); } break; - case 'P' : - status = ERROR; - LOG_ERROR("Untyped pointer are not supported for serialization"); - break; - case 'l': - status = jsonSerializer_parseAny(type->ref.ref, loc, val); - break; + //case 'P' : default : status = ERROR; - LOG_ERROR("Error provided type '%c' not supported for JSON\n", dynType_descriptorType(type)); + celix_err_pushf("Error provided type '%c' not supported for JSON\n", c); break; } return status; } -static int jsonSerializer_parseSequence(dyn_type *seq, json_t *array, void *seqLoc) { +static int jsonSerializer_parseSequence(const dyn_type* seq, json_t* array, void* seqLoc) { assert(dynType_type(seq) == DYN_TYPE_SEQUENCE); int status = OK; size_t size = json_array_size(array); - status = dynType_sequence_alloc(seq, seqLoc, (int) size); - - if (status == OK) { - dyn_type *itemType = dynType_sequence_itemType(seq); - size_t index; - json_t *val; - json_array_foreach(array, index, val) { - void *valLoc = NULL; - status = dynType_sequence_increaseLengthAndReturnLastLoc(seq, seqLoc, &valLoc); - if (status == OK) { - status = jsonSerializer_parseAny(itemType, valLoc, val); - if (status != OK) { - break; - } - } + if (size > UINT32_MAX) { + celix_err_pushf("Error array size(%zu) too large", size); + return ERROR; + } + if ((status = dynType_sequence_alloc(seq, seqLoc, (uint32_t) size)) != OK) { + return status; + } + + const dyn_type* itemType = dynType_sequence_itemType(seq); + size_t index; + json_t* val; + json_array_foreach(array, index, val) { + void* valLoc = NULL; + (void)dynType_sequence_increaseLengthAndReturnLastLoc(seq, seqLoc, &valLoc); + status = jsonSerializer_parseAny(itemType, valLoc, val); + if (status != OK) { + break; } } return status; } -int jsonSerializer_serialize(dyn_type *type, const void* input, char **output) { - int status = OK; - - json_t *root = NULL; - status = jsonSerializer_serializeJson(type, input, &root); - - if (status == OK) { - *output = json_dumps(root, JSON_COMPACT | JSON_ENCODE_ANY); - json_decref(root); +int jsonSerializer_serialize(const dyn_type* type, const void* input, char** output) { + int status; + json_auto_t* root = NULL; + if ((status = jsonSerializer_serializeJson(type, input, &root)) != OK) { + return status; } - - return status; + *output = json_dumps(root, JSON_COMPACT | JSON_ENCODE_ANY); + return *output != NULL ? OK : ERROR; } -static int jsonSerializer_parseEnum(dyn_type *type, const char* enum_name, int32_t *out) { - struct meta_entry * entry; +static int jsonSerializer_parseEnum(const dyn_type* type, const char* enum_name, int32_t* out) { + struct meta_entry* entry; TAILQ_FOREACH(entry, &type->metaProperties, entries) { if (0 == strcmp(enum_name, entry->name)) { @@ -327,95 +260,78 @@ static int jsonSerializer_parseEnum(dyn_type *type, const char* enum_name, int32 } } - LOG_ERROR("Could not find Enum value %s in enum type", enum_name); + celix_err_pushf("Could not find Enum value %s in enum type", enum_name); return ERROR; } -int jsonSerializer_serializeJson(dyn_type *type, const void* input, json_t **out) { - return jsonSerializer_writeAny(type, (void*)input /*TODO update static function to take const void**/, out); +int jsonSerializer_serializeJson(const dyn_type* type, const void* input, json_t** out) { + return jsonSerializer_writeAny(dynType_realType(type), input, out); } -static int jsonSerializer_writeAny(dyn_type *type, void* input, json_t **out) { +static int jsonSerializer_writeAny(const dyn_type* type, const void* input, json_t** out) { int status = OK; int descriptor = dynType_descriptorType(type); - json_t *val = NULL; - dyn_type *subType = NULL; - - bool *z; //Z - float *f; //F - double *d; //D - char *b; //B - int *n; //N - int16_t *s; //S - int32_t *i; //I - int32_t *e; //E - int64_t *l; //J - uint8_t *ub; //b - uint16_t *us; //s - uint32_t *ui; //i - uint64_t *ul; //j + json_auto_t* val = NULL; + const dyn_type* subType = NULL; switch (descriptor) { case 'Z' : - z = input; - val = json_boolean((bool)*z); + val = json_boolean(*(const bool*)input); break; case 'B' : - b = input; - val = json_integer((json_int_t)*b); + val = json_integer((json_int_t)*(const char*)input); break; case 'S' : - s = input; - val = json_integer((json_int_t)*s); + val = json_integer((json_int_t)*(const int16_t*)input); break; case 'I' : - i = input; - val = json_integer((json_int_t)*i); + val = json_integer((json_int_t)*(const int32_t*)input); break; case 'J' : - l = input; - val = json_integer((json_int_t)*l); + val = json_integer((json_int_t)*(const int64_t*)input); break; case 'b' : - ub = input; - val = json_integer((json_int_t)*ub); + val = json_integer((json_int_t)*(const uint8_t*)input); break; case 's' : - us = input; - val = json_integer((json_int_t)*us); + val = json_integer((json_int_t)*(const uint16_t*)input); break; case 'i' : - ui = input; - val = json_integer((json_int_t)*ui); + val = json_integer((json_int_t)*(const uint32_t *)input); break; case 'j' : - ul = input; - val = json_integer((json_int_t)*ul); + val = json_integer((json_int_t)*(const uint64_t*)input); break; case 'N' : - n = input; - val = json_integer((json_int_t)*n); + val = json_integer((json_int_t)*(const int*)input); break; case 'F' : - f = input; - val = json_real((double) *f); + val = json_real((double) *(const float*)input); break; case 'D' : - d = input; - val = json_real(*d); + val = json_real(*(const double*)input); break; - case 't' : - val = json_string(*(const char **) input); + case 't' : { + const char *strValue = *(const char **) input; + val = (strValue != NULL) ? json_string(strValue) : json_null(); break; + } case 'E': - e = input; - status = jsonSerializer_writeEnum(type, *e, &val); + status = jsonSerializer_writeEnum(type, *(const int32_t*)input, &val); break; case '*' : - status = dynType_typedPointer_getTypedType(type, &subType); - if (status == OK) { - status = jsonSerializer_writeAny(subType, *(void **)input, &val); + subType = dynType_typedPointer_getTypedType(type); + if (dynType_ffiType(subType) != &ffi_type_pointer) { + const void* inputValue = *(const void**)input; + if (inputValue) { + status = jsonSerializer_writeAny(subType, inputValue, &val); + } else { + val = json_null(); + } + } else { + status = ERROR; + celix_err_pushf("Error cannot serialize pointer to pointer"); } break; case '{' : @@ -424,114 +340,84 @@ static int jsonSerializer_writeAny(dyn_type *type, void* input, json_t **out) { case '[' : status = jsonSerializer_writeSequence(type, input, &val); break; - case 'P' : - status = ERROR; - LOG_ERROR("Untyped pointer not supported for serialization."); - break; - case 'l': - status = jsonSerializer_writeAny(type->ref.ref, input, out); - break; default : - LOG_ERROR("Unsupported descriptor '%c'", descriptor); + celix_err_pushf("Unsupported descriptor '%c'", descriptor); status = ERROR; break; } - - if (status == OK && val != NULL) { - *out = val; + if (status != OK) { + return status; } - - return status; + *out = celix_steal_ptr(val); + return *out != NULL ? OK : ERROR; } -static int jsonSerializer_writeSequence(dyn_type *type, void *input, json_t **out) { +static int jsonSerializer_writeSequence(const dyn_type* type, const void* input, json_t** out) { assert(dynType_type(type) == DYN_TYPE_SEQUENCE); - int status = OK; - json_t *array = json_array(); - dyn_type *itemType = dynType_sequence_itemType(type); + json_auto_t* array = json_array(); + if (array == NULL) { + return ERROR; + } + const dyn_type* itemType = dynType_sequence_itemType(type); uint32_t len = dynType_sequence_length(input); - int i = 0; - void *itemLoc = NULL; - json_t *item = NULL; - for (i = 0; i < len; i += 1) { - item = NULL; - status = dynType_sequence_locForIndex(type, input, i, &itemLoc); - if (status == OK) { - status = jsonSerializer_writeAny(itemType, itemLoc, &item); - if (status == OK) { - json_array_append(array, item); - json_decref(item); - } + for (uint32_t i = 0; i < len; i += 1) { + int status = OK; + void* itemLoc = NULL; + json_t* item = NULL; + if ((status = dynType_sequence_locForIndex(type, input, i, &itemLoc)) != OK) { + celix_err_push("Cannot serialize invalid sequence"); + return status; } - - if (status != OK) { - break; + if ((status = jsonSerializer_writeAny(itemType, itemLoc, &item)) != OK) { + return status; + } + if ((json_array_append_new(array, item)) != 0) { + return ERROR; } } - if (status == OK && array != NULL) { - *out = array; - } else { - *out = NULL; - json_decref(array); - } - - return status; + *out = celix_steal_ptr(array); + return OK; } -static int jsonSerializer_writeComplex(dyn_type *type, void *input, json_t **out) { +static int jsonSerializer_writeComplex(const dyn_type* type, const void* input, json_t** out) { assert(dynType_type(type) == DYN_TYPE_COMPLEX); - int status = OK; - json_t *val = json_object(); - struct complex_type_entry *entry = NULL; - struct complex_type_entries_head *entries = NULL; - int index = -1; - - status = dynType_complex_entries(type, &entries); - if (status == OK) { - TAILQ_FOREACH(entry, entries, entries) { - void *subLoc = NULL; - json_t *subVal = NULL; - dyn_type *subType = NULL; - index = dynType_complex_indexForName(type, entry->name); - if (index < 0) { - LOG_ERROR("Cannot find index for member '%s'", entry->name); - status = ERROR; - } - if(status == OK){ - status = dynType_complex_valLocAt(type, index, input, &subLoc); - } - if (status == OK) { - status = dynType_complex_dynTypeAt(type, index, &subType); - } - if (status == OK) { - status = jsonSerializer_writeAny(subType, subLoc, &subVal); - } - if (status == OK) { - json_object_set(val, entry->name, subVal); - json_decref(subVal); - } - - if (status != OK) { - break; - } - } + json_auto_t* val = json_object(); + if (val == NULL) { + return ERROR; } - - if (status == OK && val != NULL) { - *out = val; - } else { - *out = NULL; - json_decref(val); + struct complex_type_entry* entry = NULL; + const struct complex_type_entries_head* entries = dynType_complex_entries(type); + int index = 0; + + TAILQ_FOREACH(entry, entries, entries) { + int status; + void* subLoc = NULL; + json_t* subVal = NULL; + const dyn_type* subType = NULL; + if (entry->name == NULL) { + celix_err_push("Unamed field unsupported"); + return ERROR; + } + subLoc = dynType_complex_valLocAt(type, index, (void*)input); + subType = dynType_complex_dynTypeAt(type, index); + if ((status = jsonSerializer_writeAny(subType, subLoc, &subVal)) != OK) { + return status; + } + if (json_object_set_new(val, entry->name, subVal) != 0) { + return ERROR; + } + index++; } - return status; + *out = celix_steal_ptr(val); + return OK; } -static int jsonSerializer_writeEnum(dyn_type *type, int32_t enum_value, json_t **out) { +static int jsonSerializer_writeEnum(const dyn_type* type, int32_t enum_value, json_t **out) { struct meta_entry * entry; // Convert to string @@ -542,10 +428,10 @@ static int jsonSerializer_writeEnum(dyn_type *type, int32_t enum_value, json_t * TAILQ_FOREACH(entry, &type->metaProperties, entries) { if (0 == strcmp(enum_value_str, entry->value)) { *out = json_string((const char*)entry->name); - return OK; + return *out != NULL ? OK : ERROR; } } - LOG_ERROR("Could not find Enum value %s in enum type", enum_value_str); + celix_err_pushf("Could not find Enum value %s in enum type", enum_value_str); return ERROR; } diff --git a/libs/error_injector/CMakeLists.txt b/libs/error_injector/CMakeLists.txt index 958d99c94..76a9f99f7 100644 --- a/libs/error_injector/CMakeLists.txt +++ b/libs/error_injector/CMakeLists.txt @@ -37,7 +37,13 @@ add_subdirectory(sys_shm) add_subdirectory(socket) add_subdirectory(pthread) add_subdirectory(unistd) + celix_subproject(ERROR_INJECTOR_MDNSRESPONDER "Option to enable building the mdnsresponder error injector" OFF) if (ERROR_INJECTOR_MDNSRESPONDER) add_subdirectory(mdnsresponder) endif () + +celix_subproject(ERROR_INJECTOR_JANSSON "Option to enable building the jansson error injector" ON) +if (ERROR_INJECTOR_JANSSON) + add_subdirectory(jansson) +endif () diff --git a/libs/error_injector/jansson/CMakeLists.txt b/libs/error_injector/jansson/CMakeLists.txt new file mode 100644 index 000000000..b536956a0 --- /dev/null +++ b/libs/error_injector/jansson/CMakeLists.txt @@ -0,0 +1,39 @@ +# 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. + +find_package(jansson REQUIRED) + +add_library(jansson_ei STATIC src/jansson_ei.cc) + +target_include_directories(jansson_ei PUBLIC ${CMAKE_CURRENT_LIST_DIR}/include) +target_link_libraries(jansson_ei + PUBLIC Celix::error_injector jansson::jansson + PRIVATE Celix::utils #for celix_cleanup.h +) + +target_link_options(jansson_ei INTERFACE + LINKER:--wrap,json_array_size + LINKER:--wrap,json_dumps + LINKER:--wrap,json_object + LINKER:--wrap,json_object_set_new + LINKER:--wrap,json_array + LINKER:--wrap,json_array_append_new + LINKER:--wrap,json_integer + LINKER:--wrap,json_string + LINKER:--wrap,json_real + ) +add_library(Celix::jansson_ei ALIAS jansson_ei) diff --git a/libs/error_injector/jansson/include/jansson_ei.h b/libs/error_injector/jansson/include/jansson_ei.h new file mode 100644 index 000000000..60f04e45d --- /dev/null +++ b/libs/error_injector/jansson/include/jansson_ei.h @@ -0,0 +1,41 @@ +/* + 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. + */ +#ifndef CELIX_JANSSON_EI_H +#define CELIX_JANSSON_EI_H +#ifdef __cplusplus +extern "C" { +#endif + +#include +#include "celix_error_injector.h" + +CELIX_EI_DECLARE(json_array_size, size_t); +CELIX_EI_DECLARE(json_dumps, char*); +CELIX_EI_DECLARE(json_object, json_t*); +CELIX_EI_DECLARE(json_object_set_new, int); +CELIX_EI_DECLARE(json_array, json_t*); +CELIX_EI_DECLARE(json_array_append_new, int); +CELIX_EI_DECLARE(json_integer, json_t*); +CELIX_EI_DECLARE(json_string, json_t*); +CELIX_EI_DECLARE(json_real, json_t*); + +#ifdef __cplusplus +} +#endif +#endif //CELIX_JANSSON_EI_H diff --git a/libs/error_injector/jansson/src/jansson_ei.cc b/libs/error_injector/jansson/src/jansson_ei.cc new file mode 100644 index 000000000..57033c0c4 --- /dev/null +++ b/libs/error_injector/jansson/src/jansson_ei.cc @@ -0,0 +1,91 @@ +/* + 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. + */ + +#include "jansson_ei.h" +#include "celix_cleanup.h" +#include + +extern "C" { + +size_t __real_json_array_size(const json_t *array); +CELIX_EI_DEFINE(json_array_size, size_t) +size_t __wrap_json_array_size(const json_t *array) { + CELIX_EI_IMPL(json_array_size); + return __real_json_array_size(array); +} + +char *__real_json_dumps(const json_t *json, size_t flags); +CELIX_EI_DEFINE(json_dumps, char*) +char *__wrap_json_dumps(const json_t *json, size_t flags) { + CELIX_EI_IMPL(json_dumps); + return __real_json_dumps(json, flags); +} + +json_t *__real_json_object(void); +CELIX_EI_DEFINE(json_object, json_t*) +json_t *__wrap_json_object(void) { + CELIX_EI_IMPL(json_object); + return __real_json_object(); +} + +int __real_json_object_set_new(json_t *object, const char *key, json_t *value); +CELIX_EI_DEFINE(json_object_set_new, int) +int __wrap_json_object_set_new(json_t *object, const char *key, json_t *value) { + json_auto_t *val = value; + CELIX_EI_IMPL(json_object_set_new); + return __real_json_object_set_new(object, key, celix_steal_ptr(val)); +} + +json_t *__real_json_array(void); +CELIX_EI_DEFINE(json_array, json_t*) +json_t *__wrap_json_array(void) { + CELIX_EI_IMPL(json_array); + return __real_json_array(); +} + +int __real_json_array_append_new(json_t *array, json_t *value); +CELIX_EI_DEFINE(json_array_append_new, int) +int __wrap_json_array_append_new(json_t *array, json_t *value) { + json_auto_t *val = value; + CELIX_EI_IMPL(json_array_append_new); + return __real_json_array_append_new(array, celix_steal_ptr(val)); +} + +json_t *__real_json_integer(json_int_t value); +CELIX_EI_DEFINE(json_integer, json_t*) +json_t *__wrap_json_integer(json_int_t value) { + CELIX_EI_IMPL(json_integer); + return __real_json_integer(value); +} + +json_t *__real_json_string(const char *value); +CELIX_EI_DEFINE(json_string, json_t*) +json_t *__wrap_json_string(const char *value) { + CELIX_EI_IMPL(json_string); + return __real_json_string(value); +} + +json_t *__real_json_real(double value); +CELIX_EI_DEFINE(json_real, json_t*) +json_t *__wrap_json_real(double value) { + CELIX_EI_IMPL(json_real); + return __real_json_real(value); +} + +} \ No newline at end of file diff --git a/libs/error_injector/stdio/CMakeLists.txt b/libs/error_injector/stdio/CMakeLists.txt index f4824197e..2e53d8099 100644 --- a/libs/error_injector/stdio/CMakeLists.txt +++ b/libs/error_injector/stdio/CMakeLists.txt @@ -31,5 +31,7 @@ target_link_options(stdio_ei INTERFACE LINKER:--wrap,fputc LINKER:--wrap,fputs LINKER:--wrap,fclose + LINKER:--wrap,fgetc + LINKER:--wrap,fmemopen ) add_library(Celix::stdio_ei ALIAS stdio_ei) diff --git a/libs/error_injector/stdio/include/stdio_ei.h b/libs/error_injector/stdio/include/stdio_ei.h index 321a39d9b..b5be1e311 100644 --- a/libs/error_injector/stdio/include/stdio_ei.h +++ b/libs/error_injector/stdio/include/stdio_ei.h @@ -26,13 +26,13 @@ extern "C" { #include "celix_error_injector.h" #include -CELIX_EI_DECLARE(fopen, FILE *); +CELIX_EI_DECLARE(fopen, FILE*); CELIX_EI_DECLARE(fwrite, size_t); CELIX_EI_DECLARE(remove, int); -CELIX_EI_DECLARE(open_memstream, FILE *); +CELIX_EI_DECLARE(open_memstream, FILE*); CELIX_EI_DECLARE(fseek, int); @@ -46,6 +46,10 @@ CELIX_EI_DECLARE(fputs, int); CELIX_EI_DECLARE(fclose, int); +CELIX_EI_DECLARE(fgetc, int); + +CELIX_EI_DECLARE(fmemopen, FILE*); + #ifdef __cplusplus } #endif diff --git a/libs/error_injector/stdio/src/stdio_ei.cc b/libs/error_injector/stdio/src/stdio_ei.cc index d78f7bbc2..f9bc89d52 100644 --- a/libs/error_injector/stdio/src/stdio_ei.cc +++ b/libs/error_injector/stdio/src/stdio_ei.cc @@ -108,4 +108,20 @@ int __wrap_fclose(FILE* __stream) { return rc; } +int __real_fgetc(FILE* __stream); +CELIX_EI_DEFINE(fgetc, int) +int __wrap_fgetc(FILE* __stream) { + CELIX_EI_IMPL(fgetc); + return __real_fgetc(__stream); } + +FILE* __real_fmemopen(void* __s, size_t __len, const char* __modes); +CELIX_EI_DEFINE(fmemopen, FILE*) +FILE* __wrap_fmemopen(void* __s, size_t __len, const char* __modes) { + errno = ENOMEM; + CELIX_EI_IMPL(fmemopen); + errno = 0; + return __real_fmemopen(__s, __len, __modes); +} + +} \ No newline at end of file diff --git a/libs/error_injector/string/CMakeLists.txt b/libs/error_injector/string/CMakeLists.txt index ff53b52d3..190048570 100644 --- a/libs/error_injector/string/CMakeLists.txt +++ b/libs/error_injector/string/CMakeLists.txt @@ -22,5 +22,6 @@ target_link_libraries(string_ei PUBLIC Celix::error_injector) target_link_options(string_ei INTERFACE LINKER:--wrap,strndup + LINKER:--wrap,strdup ) add_library(Celix::string_ei ALIAS string_ei) diff --git a/libs/error_injector/string/include/string_ei.h b/libs/error_injector/string/include/string_ei.h index 85a2b3543..0ef7d9c83 100644 --- a/libs/error_injector/string/include/string_ei.h +++ b/libs/error_injector/string/include/string_ei.h @@ -29,6 +29,8 @@ extern "C" { CELIX_EI_DECLARE(strndup, char *); +CELIX_EI_DECLARE(strdup, char *); + #ifdef __cplusplus } #endif diff --git a/libs/error_injector/string/src/string_ei.cc b/libs/error_injector/string/src/string_ei.cc index 747478924..1b26f5ccd 100644 --- a/libs/error_injector/string/src/string_ei.cc +++ b/libs/error_injector/string/src/string_ei.cc @@ -22,12 +22,22 @@ #include extern "C" { -char* __real_strndup(const char* s, size_t n); +char *__real_strndup(const char *s, size_t n); CELIX_EI_DEFINE(strndup, char*) -char* __wrap_strndup(const char* s, size_t n) { +char *__wrap_strndup(const char *s, size_t n) { errno = ENOMEM; CELIX_EI_IMPL(strndup); errno = 0; return __real_strndup(s, n); } + +char *__real_strdup(const char *s); +CELIX_EI_DEFINE(strdup, char*) +char *__wrap_strdup(const char *s) { + errno = ENOMEM; + CELIX_EI_IMPL(strdup); + errno = 0; + return __real_strdup(s); +} + } \ No newline at end of file diff --git a/libs/framework/gtest/src/ManifestTestSuite.cc b/libs/framework/gtest/src/ManifestTestSuite.cc index 6cb668353..7c96903be 100644 --- a/libs/framework/gtest/src/ManifestTestSuite.cc +++ b/libs/framework/gtest/src/ManifestTestSuite.cc @@ -21,10 +21,6 @@ #include #include #include -#if CELIX_UTILS_NO_MEMSTREAM_AVAILABLE -#include "open_memstream.h" -#include "fmemopen.h" -#endif #include "celix_err.h" #include "celix_stdio_cleanup.h" diff --git a/libs/framework/src/celix_log.c b/libs/framework/src/celix_log.c index d765ee9e5..c8606b43a 100644 --- a/libs/framework/src/celix_log.c +++ b/libs/framework/src/celix_log.c @@ -28,10 +28,6 @@ #include "celix_threads.h" #include "celix_array_list.h" -#ifdef CELIX_UTILS_NO_MEMSTREAM_AVAILABLE -#include "memstream/open_memstream.h" -#endif - #define LOG_NAME "celix_framework" struct celix_framework_logger { diff --git a/libs/utils/CMakeLists.txt b/libs/utils/CMakeLists.txt index 0f4dd6418..e1b498359 100644 --- a/libs/utils/CMakeLists.txt +++ b/libs/utils/CMakeLists.txt @@ -21,14 +21,6 @@ if (UTILS) set(MEMSTREAM_SOURCES ) set(MEMSTREAM_INCLUDES ) - include(CheckFunctionExists) - #check_function_exists(fmemopen FMEMOPEN_EXISTS) - check_function_exists(open_memstream OPEN_MEMSTREAM_EXISTS) - if (NOT OPEN_MEMSTREAM_EXISTS) - set(MEMSTREAM_SOURCES src/memstream/open_memstream.c src/memstream/fmemopen.c ${MEMSTREAM_SOURCES}) - set(MEMSTREAM_INCLUDE_DIR ${CMAKE_CURRENT_LIST_DIR}/include/memstream ${MEMSTREAM_INCLUDE_DIR}) - install(DIRECTORY include/memstream/ DESTINATION ${CMAKE_INSTALL_INCLUDEDIR}/celix/memstream COMPONENT framework) - endif() set(UTILS_SRC src/array_list.c @@ -60,10 +52,6 @@ if (UTILS) OUTPUT_NAME "celix_utils") celix_target_hide_symbols(utils) - if (NOT OPEN_MEMSTREAM_EXISTS) - target_compile_definitions(utils PUBLIC -DCELIX_UTILS_NO_MEMSTREAM_AVAILABLE) - endif () - if (ANDROID) target_compile_definitions(utils PRIVATE -DUSE_FILE32API) endif () @@ -75,7 +63,6 @@ if (UTILS) target_include_directories(utils PUBLIC $ $ #Note not installed, only for project internal usage - $ ) target_include_directories(utils PRIVATE src include_deprecated) IF(UNIX AND NOT ANDROID) @@ -126,7 +113,6 @@ if (UTILS) target_include_directories(utils_cut PUBLIC ${CMAKE_CURRENT_LIST_DIR}/include ${CMAKE_CURRENT_LIST_DIR}/include_internal - ${MEMSTREAM_INCLUDE_DIR} ${CMAKE_BINARY_DIR}/celix/gen/includes/utils ${CMAKE_BINARY_DIR}/celix/gen/src/utils src include_deprecated diff --git a/libs/utils/include_deprecated/celix_utils_api.h b/libs/utils/include_deprecated/celix_utils_api.h index 42779c01c..8a42325c1 100644 --- a/libs/utils/include_deprecated/celix_utils_api.h +++ b/libs/utils/include_deprecated/celix_utils_api.h @@ -31,9 +31,4 @@ #include "celix_utils.h" #include "celix_version.h" -#if defined(NO_MEMSTREAM_AVAILABLE) -#include "memstream/open_memstream.h" -#include "memstream/fmemopen.h" -#endif - #endif //CELIX_CELIX_UTILS_API_H_ diff --git a/libs/utils/src/memstream/fmemopen.c b/libs/utils/src/memstream/fmemopen.c deleted file mode 100644 index cb1b0c068..000000000 --- a/libs/utils/src/memstream/fmemopen.c +++ /dev/null @@ -1,76 +0,0 @@ - -/* - * fmem.c : fmemopen() on top of BSD's funopen() - * 20081017 AF - */ - -#include -#include -#include - -struct fmem { - size_t pos; - size_t size; - char *buffer; -}; -typedef struct fmem fmem_t; - -static int readfn(void *handler, char *buf, int size) -{ - int count = 0; - fmem_t *mem = handler; - size_t available = mem->size - mem->pos; - - if(size > available) size = available; - for(count=0; count < size; mem->pos++, count++) - buf[count] = mem->buffer[mem->pos]; - - return count; -} - -static int writefn(void *handler, const char *buf, int size) -{ - int count = 0; - fmem_t *mem = handler; - size_t available = mem->size - mem->pos; - - if(size > available) size = available; - for(count=0; count < size; mem->pos++, count++) - mem->buffer[mem->pos] = buf[count]; - - return count; // ? count : size; -} - -static fpos_t seekfn(void *handler, fpos_t offset, int whence) -{ - size_t pos; - fmem_t *mem = handler; - - switch(whence) { - case SEEK_SET: pos = offset; break; - case SEEK_CUR: pos = mem->pos + offset; break; - case SEEK_END: pos = mem->size + offset; break; - default: return -1; - } - - if(pos > mem->size) return -1; - - mem->pos = pos; - return (fpos_t) pos; -} - -static int closefn(void *handler) -{ - free(handler); - return 0; -} - -/* simple, but portable version of fmemopen for OS X / BSD */ -FILE *fmemopen(void *buf, size_t size, const char *mode) -{ - fmem_t *mem = (fmem_t *) malloc(sizeof(fmem_t)); - - memset(mem, 0, sizeof(fmem_t)); - mem->size = size, mem->buffer = buf; - return funopen(mem, readfn, writefn, seekfn, closefn); -} diff --git a/libs/utils/src/memstream/open_memstream.c b/libs/utils/src/memstream/open_memstream.c deleted file mode 100644 index 39d62e315..000000000 --- a/libs/utils/src/memstream/open_memstream.c +++ /dev/null @@ -1,130 +0,0 @@ -/* Use funopen(3) to provide open_memstream(3) like functionality. */ - -#include -#include -#include -#include - -struct memstream { - char **cp; - size_t *lenp; - size_t offset; -}; - -static void -memstream_grow(struct memstream *ms, size_t newsize) -{ - char *buf; - - if (newsize > *ms->lenp) { - buf = realloc(*ms->cp, newsize + 1); - if (buf != NULL) { -#ifdef DEBUG - fprintf(stderr, "MS: %p growing from %zd to %zd\n", - ms, *ms->lenp, newsize); -#endif - memset(buf + *ms->lenp + 1, 0, newsize - *ms->lenp); - *ms->cp = buf; - *ms->lenp = newsize; - } - } -} - -static int -memstream_read(void *cookie, char *buf, int len) -{ - struct memstream *ms; - int tocopy; - - ms = cookie; - memstream_grow(ms, ms->offset + len); - tocopy = *ms->lenp - ms->offset; - if (len < tocopy) - tocopy = len; - memcpy(buf, *ms->cp + ms->offset, tocopy); - ms->offset += tocopy; -#ifdef DEBUG - fprintf(stderr, "MS: read(%p, %d) = %d\n", ms, len, tocopy); -#endif - return (tocopy); -} - -static int -memstream_write(void *cookie, const char *buf, int len) -{ - struct memstream *ms; - int tocopy; - - ms = cookie; - memstream_grow(ms, ms->offset + len); - tocopy = *ms->lenp - ms->offset; - if (len < tocopy) - tocopy = len; - memcpy(*ms->cp + ms->offset, buf, tocopy); - ms->offset += tocopy; -#ifdef DEBUG - fprintf(stderr, "MS: write(%p, %d) = %d\n", ms, len, tocopy); -#endif - return (tocopy); -} - -static fpos_t -memstream_seek(void *cookie, fpos_t pos, int whence) -{ - struct memstream *ms; -#ifdef DEBUG - size_t old; -#endif - - ms = cookie; -#ifdef DEBUG - old = ms->offset; -#endif - switch (whence) { - case SEEK_SET: - ms->offset = pos; - break; - case SEEK_CUR: - ms->offset += pos; - break; - case SEEK_END: - ms->offset = *ms->lenp + pos; - break; - } -#ifdef DEBUG - fprintf(stderr, "MS: seek(%p, %lld, %d) %zd -> %zd\n", ms, pos, whence, - old, ms->offset); -#endif - return (ms->offset); -} - -static int -memstream_close(void *cookie) -{ - - free(cookie); - return (0); -} - -FILE * -open_memstream(char **cp, size_t *lenp) -{ - struct memstream *ms; - int save_errno; - FILE *fp; - - *cp = NULL; - *lenp = 0; - ms = malloc(sizeof(*ms)); - ms->cp = cp; - ms->lenp = lenp; - ms->offset = 0; - fp = funopen(ms, memstream_read, memstream_write, memstream_seek, - memstream_close); - if (fp == NULL) { - save_errno = errno; - free(ms); - errno = save_errno; - } - return (fp); -} \ No newline at end of file