Drop use of the deprecated drmP.h from display/dc/os_types.h Fix all fallout after this change. Most of the fixes was adding a missing include of vmalloc.h. Signed-off-by: Sam Ravnborg <sam@ravnborg.org> Reviewed-by: Alex Deucher <alexander.deucher@amd.com> Cc: "Christian König" <christian.koenig@amd.com> Cc: "David (ChunMing) Zhou" <David1.Zhou@amd.com> Cc: David Airlie <airlied@linux.ie> Cc: Daniel Vetter <daniel@ffwll.ch> Link: https://patchwork.freedesktop.org/patch/msgid/20190609220757.10862-4-sam@ravnborg.org
		
			
				
	
	
		
			595 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			595 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * Copyright 2012-15 Advanced Micro Devices, Inc.
 | |
|  *
 | |
|  * Permission is hereby granted, free of charge, to any person obtaining a
 | |
|  * copy of this software and associated documentation files (the "Software"),
 | |
|  * to deal in the Software without restriction, including without limitation
 | |
|  * the rights to use, copy, modify, merge, publish, distribute, sublicense,
 | |
|  * and/or sell copies of the Software, and to permit persons to whom the
 | |
|  * Software is furnished to do so, subject to the following conditions:
 | |
|  *
 | |
|  * The above copyright notice and this permission notice shall be included in
 | |
|  * all copies or substantial portions of the Software.
 | |
|  *
 | |
|  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 | |
|  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 | |
|  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL
 | |
|  * THE COPYRIGHT HOLDER(S) OR AUTHOR(S) BE LIABLE FOR ANY CLAIM, DAMAGES OR
 | |
|  * OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
 | |
|  * ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
 | |
|  * OTHER DEALINGS IN THE SOFTWARE.
 | |
|  *
 | |
|  * Authors: AMD
 | |
|  *
 | |
|  */
 | |
| 
 | |
| /*
 | |
|  * Pre-requisites: headers required by header of this unit
 | |
|  */
 | |
| 
 | |
| #include <linux/slab.h>
 | |
| 
 | |
| #include "dm_services.h"
 | |
| #include "include/gpio_interface.h"
 | |
| #include "include/gpio_service_interface.h"
 | |
| #include "hw_translate.h"
 | |
| #include "hw_factory.h"
 | |
| 
 | |
| /*
 | |
|  * Header of this unit
 | |
|  */
 | |
| 
 | |
| #include "gpio_service.h"
 | |
| 
 | |
| /*
 | |
|  * Post-requisites: headers required by this unit
 | |
|  */
 | |
| 
 | |
| #include "hw_gpio.h"
 | |
| 
 | |
| /*
 | |
|  * @brief
 | |
|  * Public API.
 | |
|  */
 | |
| 
 | |
| struct gpio_service *dal_gpio_service_create(
 | |
| 	enum dce_version dce_version_major,
 | |
| 	enum dce_version dce_version_minor,
 | |
| 	struct dc_context *ctx)
 | |
| {
 | |
| 	struct gpio_service *service;
 | |
| 	uint32_t index_of_id;
 | |
| 
 | |
| 	service = kzalloc(sizeof(struct gpio_service), GFP_KERNEL);
 | |
| 
 | |
| 	if (!service) {
 | |
| 		BREAK_TO_DEBUGGER();
 | |
| 		return NULL;
 | |
| 	}
 | |
| 
 | |
| 	if (!dal_hw_translate_init(&service->translate, dce_version_major,
 | |
| 			dce_version_minor)) {
 | |
| 		BREAK_TO_DEBUGGER();
 | |
| 		goto failure_1;
 | |
| 	}
 | |
| 
 | |
| 	if (!dal_hw_factory_init(&service->factory, dce_version_major,
 | |
| 			dce_version_minor)) {
 | |
| 		BREAK_TO_DEBUGGER();
 | |
| 		goto failure_1;
 | |
| 	}
 | |
| 
 | |
| 	/* allocate and initialize busyness storage */
 | |
| 	{
 | |
| 		index_of_id = 0;
 | |
| 		service->ctx = ctx;
 | |
| 
 | |
| 		do {
 | |
| 			uint32_t number_of_bits =
 | |
| 				service->factory.number_of_pins[index_of_id];
 | |
| 			uint32_t i = 0;
 | |
| 
 | |
| 			if (number_of_bits)  {
 | |
| 				service->busyness[index_of_id] =
 | |
| 					kcalloc(number_of_bits, sizeof(char),
 | |
| 						GFP_KERNEL);
 | |
| 
 | |
| 				if (!service->busyness[index_of_id]) {
 | |
| 					BREAK_TO_DEBUGGER();
 | |
| 					goto failure_2;
 | |
| 				}
 | |
| 
 | |
| 				do {
 | |
| 					service->busyness[index_of_id][i] = 0;
 | |
| 					++i;
 | |
| 				} while (i < number_of_bits);
 | |
| 			} else {
 | |
| 				service->busyness[index_of_id] = NULL;
 | |
| 			}
 | |
| 
 | |
| 			++index_of_id;
 | |
| 		} while (index_of_id < GPIO_ID_COUNT);
 | |
| 	}
 | |
| 
 | |
| 	return service;
 | |
| 
 | |
| failure_2:
 | |
| 	while (index_of_id) {
 | |
| 		--index_of_id;
 | |
| 		kfree(service->busyness[index_of_id]);
 | |
| 	}
 | |
| 
 | |
| failure_1:
 | |
| 	kfree(service);
 | |
| 
 | |
| 	return NULL;
 | |
| }
 | |
| 
 | |
| struct gpio *dal_gpio_service_create_irq(
 | |
| 	struct gpio_service *service,
 | |
| 	uint32_t offset,
 | |
| 	uint32_t mask)
 | |
| {
 | |
| 	enum gpio_id id;
 | |
| 	uint32_t en;
 | |
| 
 | |
| 	if (!service->translate.funcs->offset_to_id(offset, mask, &id, &en)) {
 | |
| 		ASSERT_CRITICAL(false);
 | |
| 		return NULL;
 | |
| 	}
 | |
| 
 | |
| 	return dal_gpio_create_irq(service, id, en);
 | |
| }
 | |
| 
 | |
| void dal_gpio_service_destroy(
 | |
| 	struct gpio_service **ptr)
 | |
| {
 | |
| 	if (!ptr || !*ptr) {
 | |
| 		BREAK_TO_DEBUGGER();
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	/* free business storage */
 | |
| 	{
 | |
| 		uint32_t index_of_id = 0;
 | |
| 
 | |
| 		do {
 | |
| 			kfree((*ptr)->busyness[index_of_id]);
 | |
| 
 | |
| 			++index_of_id;
 | |
| 		} while (index_of_id < GPIO_ID_COUNT);
 | |
| 	}
 | |
| 
 | |
| 	kfree(*ptr);
 | |
| 
 | |
| 	*ptr = NULL;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * @brief
 | |
|  * Private API.
 | |
|  */
 | |
| 
 | |
| static bool is_pin_busy(
 | |
| 	const struct gpio_service *service,
 | |
| 	enum gpio_id id,
 | |
| 	uint32_t en)
 | |
| {
 | |
| 	return service->busyness[id][en];
 | |
| }
 | |
| 
 | |
| static void set_pin_busy(
 | |
| 	struct gpio_service *service,
 | |
| 	enum gpio_id id,
 | |
| 	uint32_t en)
 | |
| {
 | |
| 	service->busyness[id][en] = true;
 | |
| }
 | |
| 
 | |
| static void set_pin_free(
 | |
| 	struct gpio_service *service,
 | |
| 	enum gpio_id id,
 | |
| 	uint32_t en)
 | |
| {
 | |
| 	service->busyness[id][en] = false;
 | |
| }
 | |
| 
 | |
| enum gpio_result dal_gpio_service_lock(
 | |
| 	struct gpio_service *service,
 | |
| 	enum gpio_id id,
 | |
| 	uint32_t en)
 | |
| {
 | |
| 	if (!service->busyness[id]) {
 | |
| 		ASSERT_CRITICAL(false);
 | |
| 		return GPIO_RESULT_OPEN_FAILED;
 | |
| 	}
 | |
| 
 | |
| 	set_pin_busy(service, id, en);
 | |
| 	return GPIO_RESULT_OK;
 | |
| }
 | |
| 
 | |
| enum gpio_result dal_gpio_service_unlock(
 | |
| 	struct gpio_service *service,
 | |
| 	enum gpio_id id,
 | |
| 	uint32_t en)
 | |
| {
 | |
| 	if (!service->busyness[id]) {
 | |
| 		ASSERT_CRITICAL(false);
 | |
| 		return GPIO_RESULT_OPEN_FAILED;
 | |
| 	}
 | |
| 
 | |
| 	set_pin_free(service, id, en);
 | |
| 	return GPIO_RESULT_OK;
 | |
| }
 | |
| 
 | |
| enum gpio_result dal_gpio_service_open(
 | |
| 	struct gpio_service *service,
 | |
| 	enum gpio_id id,
 | |
| 	uint32_t en,
 | |
| 	enum gpio_mode mode,
 | |
| 	struct hw_gpio_pin **ptr)
 | |
| {
 | |
| 	struct hw_gpio_pin *pin;
 | |
| 
 | |
| 	if (!service->busyness[id]) {
 | |
| 		ASSERT_CRITICAL(false);
 | |
| 		return GPIO_RESULT_OPEN_FAILED;
 | |
| 	}
 | |
| 
 | |
| 	if (is_pin_busy(service, id, en)) {
 | |
| 		ASSERT_CRITICAL(false);
 | |
| 		return GPIO_RESULT_DEVICE_BUSY;
 | |
| 	}
 | |
| 
 | |
| 	switch (id) {
 | |
| 	case GPIO_ID_DDC_DATA:
 | |
| 		pin = service->factory.funcs->create_ddc_data(
 | |
| 			service->ctx, id, en);
 | |
| 		service->factory.funcs->define_ddc_registers(pin, en);
 | |
| 	break;
 | |
| 	case GPIO_ID_DDC_CLOCK:
 | |
| 		pin = service->factory.funcs->create_ddc_clock(
 | |
| 			service->ctx, id, en);
 | |
| 		service->factory.funcs->define_ddc_registers(pin, en);
 | |
| 	break;
 | |
| 	case GPIO_ID_GENERIC:
 | |
| 		pin = service->factory.funcs->create_generic(
 | |
| 			service->ctx, id, en);
 | |
| 	break;
 | |
| 	case GPIO_ID_HPD:
 | |
| 		pin = service->factory.funcs->create_hpd(
 | |
| 			service->ctx, id, en);
 | |
| 		service->factory.funcs->define_hpd_registers(pin, en);
 | |
| 	break;
 | |
| 	case GPIO_ID_SYNC:
 | |
| 		pin = service->factory.funcs->create_sync(
 | |
| 			service->ctx, id, en);
 | |
| 	break;
 | |
| 	case GPIO_ID_GSL:
 | |
| 		pin = service->factory.funcs->create_gsl(
 | |
| 			service->ctx, id, en);
 | |
| 	break;
 | |
| 	default:
 | |
| 		ASSERT_CRITICAL(false);
 | |
| 		return GPIO_RESULT_NON_SPECIFIC_ERROR;
 | |
| 	}
 | |
| 
 | |
| 	if (!pin) {
 | |
| 		ASSERT_CRITICAL(false);
 | |
| 		return GPIO_RESULT_NON_SPECIFIC_ERROR;
 | |
| 	}
 | |
| 
 | |
| 	if (!pin->funcs->open(pin, mode)) {
 | |
| 		ASSERT_CRITICAL(false);
 | |
| 		dal_gpio_service_close(service, &pin);
 | |
| 		return GPIO_RESULT_OPEN_FAILED;
 | |
| 	}
 | |
| 
 | |
| 	set_pin_busy(service, id, en);
 | |
| 	*ptr = pin;
 | |
| 	return GPIO_RESULT_OK;
 | |
| }
 | |
| 
 | |
| void dal_gpio_service_close(
 | |
| 	struct gpio_service *service,
 | |
| 	struct hw_gpio_pin **ptr)
 | |
| {
 | |
| 	struct hw_gpio_pin *pin;
 | |
| 
 | |
| 	if (!ptr) {
 | |
| 		ASSERT_CRITICAL(false);
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	pin = *ptr;
 | |
| 
 | |
| 	if (pin) {
 | |
| 		set_pin_free(service, pin->id, pin->en);
 | |
| 
 | |
| 		pin->funcs->close(pin);
 | |
| 
 | |
| 		pin->funcs->destroy(ptr);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| 
 | |
| enum dc_irq_source dal_irq_get_source(
 | |
| 	const struct gpio *irq)
 | |
| {
 | |
| 	enum gpio_id id = dal_gpio_get_id(irq);
 | |
| 
 | |
| 	switch (id) {
 | |
| 	case GPIO_ID_HPD:
 | |
| 		return (enum dc_irq_source)(DC_IRQ_SOURCE_HPD1 +
 | |
| 			dal_gpio_get_enum(irq));
 | |
| 	case GPIO_ID_GPIO_PAD:
 | |
| 		return (enum dc_irq_source)(DC_IRQ_SOURCE_GPIOPAD0 +
 | |
| 			dal_gpio_get_enum(irq));
 | |
| 	default:
 | |
| 		return DC_IRQ_SOURCE_INVALID;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| enum dc_irq_source dal_irq_get_rx_source(
 | |
| 	const struct gpio *irq)
 | |
| {
 | |
| 	enum gpio_id id = dal_gpio_get_id(irq);
 | |
| 
 | |
| 	switch (id) {
 | |
| 	case GPIO_ID_HPD:
 | |
| 		return (enum dc_irq_source)(DC_IRQ_SOURCE_HPD1RX +
 | |
| 			dal_gpio_get_enum(irq));
 | |
| 	default:
 | |
| 		return DC_IRQ_SOURCE_INVALID;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| enum gpio_result dal_irq_setup_hpd_filter(
 | |
| 	struct gpio *irq,
 | |
| 	struct gpio_hpd_config *config)
 | |
| {
 | |
| 	struct gpio_config_data config_data;
 | |
| 
 | |
| 	if (!config)
 | |
| 		return GPIO_RESULT_INVALID_DATA;
 | |
| 
 | |
| 	config_data.type = GPIO_CONFIG_TYPE_HPD;
 | |
| 	config_data.config.hpd = *config;
 | |
| 
 | |
| 	return dal_gpio_set_config(irq, &config_data);
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * @brief
 | |
|  * Creation and destruction
 | |
|  */
 | |
| 
 | |
| struct gpio *dal_gpio_create_irq(
 | |
| 	struct gpio_service *service,
 | |
| 	enum gpio_id id,
 | |
| 	uint32_t en)
 | |
| {
 | |
| 	struct gpio *irq;
 | |
| 
 | |
| 	switch (id) {
 | |
| 	case GPIO_ID_HPD:
 | |
| 	case GPIO_ID_GPIO_PAD:
 | |
| 	break;
 | |
| 	default:
 | |
| 		id = GPIO_ID_HPD;
 | |
| 		ASSERT_CRITICAL(false);
 | |
| 		return NULL;
 | |
| 	}
 | |
| 
 | |
| 	irq = dal_gpio_create(
 | |
| 		service, id, en, GPIO_PIN_OUTPUT_STATE_DEFAULT);
 | |
| 
 | |
| 	if (irq)
 | |
| 		return irq;
 | |
| 
 | |
| 	ASSERT_CRITICAL(false);
 | |
| 	return NULL;
 | |
| }
 | |
| 
 | |
| void dal_gpio_destroy_irq(
 | |
| 	struct gpio **irq)
 | |
| {
 | |
| 	if (!irq || !*irq) {
 | |
| 		ASSERT_CRITICAL(false);
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	dal_gpio_close(*irq);
 | |
| 	dal_gpio_destroy(irq);
 | |
| 	kfree(*irq);
 | |
| 
 | |
| 	*irq = NULL;
 | |
| }
 | |
| 
 | |
| struct ddc *dal_gpio_create_ddc(
 | |
| 	struct gpio_service *service,
 | |
| 	uint32_t offset,
 | |
| 	uint32_t mask,
 | |
| 	struct gpio_ddc_hw_info *info)
 | |
| {
 | |
| 	enum gpio_id id;
 | |
| 	uint32_t en;
 | |
| 	struct ddc *ddc;
 | |
| 
 | |
| 	if (!service->translate.funcs->offset_to_id(offset, mask, &id, &en))
 | |
| 		return NULL;
 | |
| 
 | |
| 	ddc = kzalloc(sizeof(struct ddc), GFP_KERNEL);
 | |
| 
 | |
| 	if (!ddc) {
 | |
| 		BREAK_TO_DEBUGGER();
 | |
| 		return NULL;
 | |
| 	}
 | |
| 
 | |
| 	ddc->pin_data = dal_gpio_create(
 | |
| 		service, GPIO_ID_DDC_DATA, en, GPIO_PIN_OUTPUT_STATE_DEFAULT);
 | |
| 
 | |
| 	if (!ddc->pin_data) {
 | |
| 		BREAK_TO_DEBUGGER();
 | |
| 		goto failure_1;
 | |
| 	}
 | |
| 
 | |
| 	ddc->pin_clock = dal_gpio_create(
 | |
| 		service, GPIO_ID_DDC_CLOCK, en, GPIO_PIN_OUTPUT_STATE_DEFAULT);
 | |
| 
 | |
| 	if (!ddc->pin_clock) {
 | |
| 		BREAK_TO_DEBUGGER();
 | |
| 		goto failure_2;
 | |
| 	}
 | |
| 
 | |
| 	ddc->hw_info = *info;
 | |
| 
 | |
| 	ddc->ctx = service->ctx;
 | |
| 
 | |
| 	return ddc;
 | |
| 
 | |
| failure_2:
 | |
| 	dal_gpio_destroy(&ddc->pin_data);
 | |
| 
 | |
| failure_1:
 | |
| 	kfree(ddc);
 | |
| 
 | |
| 	return NULL;
 | |
| }
 | |
| 
 | |
| void dal_gpio_destroy_ddc(
 | |
| 	struct ddc **ddc)
 | |
| {
 | |
| 	if (!ddc || !*ddc) {
 | |
| 		BREAK_TO_DEBUGGER();
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	dal_ddc_close(*ddc);
 | |
| 	dal_gpio_destroy(&(*ddc)->pin_data);
 | |
| 	dal_gpio_destroy(&(*ddc)->pin_clock);
 | |
| 	kfree(*ddc);
 | |
| 
 | |
| 	*ddc = NULL;
 | |
| }
 | |
| 
 | |
| enum gpio_result dal_ddc_open(
 | |
| 	struct ddc *ddc,
 | |
| 	enum gpio_mode mode,
 | |
| 	enum gpio_ddc_config_type config_type)
 | |
| {
 | |
| 	enum gpio_result result;
 | |
| 
 | |
| 	struct gpio_config_data config_data;
 | |
| 	struct hw_gpio *hw_data;
 | |
| 	struct hw_gpio *hw_clock;
 | |
| 
 | |
| 	result = dal_gpio_open_ex(ddc->pin_data, mode);
 | |
| 
 | |
| 	if (result != GPIO_RESULT_OK) {
 | |
| 		BREAK_TO_DEBUGGER();
 | |
| 		return result;
 | |
| 	}
 | |
| 
 | |
| 	result = dal_gpio_open_ex(ddc->pin_clock, mode);
 | |
| 
 | |
| 	if (result != GPIO_RESULT_OK) {
 | |
| 		BREAK_TO_DEBUGGER();
 | |
| 		goto failure;
 | |
| 	}
 | |
| 
 | |
| 	/* DDC clock and data pins should belong
 | |
| 	 * to the same DDC block id,
 | |
| 	 * we use the data pin to set the pad mode. */
 | |
| 
 | |
| 	if (mode == GPIO_MODE_INPUT)
 | |
| 		/* this is from detect_sink_type,
 | |
| 		 * we need extra delay there */
 | |
| 		config_data.type = GPIO_CONFIG_TYPE_I2C_AUX_DUAL_MODE;
 | |
| 	else
 | |
| 		config_data.type = GPIO_CONFIG_TYPE_DDC;
 | |
| 
 | |
| 	config_data.config.ddc.type = config_type;
 | |
| 
 | |
| 	hw_data = FROM_HW_GPIO_PIN(ddc->pin_data->pin);
 | |
| 	hw_clock = FROM_HW_GPIO_PIN(ddc->pin_clock->pin);
 | |
| 
 | |
| 	config_data.config.ddc.data_en_bit_present = hw_data->store.en != 0;
 | |
| 	config_data.config.ddc.clock_en_bit_present = hw_clock->store.en != 0;
 | |
| 
 | |
| 	result = dal_gpio_set_config(ddc->pin_data, &config_data);
 | |
| 
 | |
| 	if (result == GPIO_RESULT_OK)
 | |
| 		return result;
 | |
| 
 | |
| 	BREAK_TO_DEBUGGER();
 | |
| 
 | |
| 	dal_gpio_close(ddc->pin_clock);
 | |
| 
 | |
| failure:
 | |
| 	dal_gpio_close(ddc->pin_data);
 | |
| 
 | |
| 	return result;
 | |
| }
 | |
| 
 | |
| enum gpio_result dal_ddc_change_mode(
 | |
| 	struct ddc *ddc,
 | |
| 	enum gpio_mode mode)
 | |
| {
 | |
| 	enum gpio_result result;
 | |
| 
 | |
| 	enum gpio_mode original_mode =
 | |
| 		dal_gpio_get_mode(ddc->pin_data);
 | |
| 
 | |
| 	result = dal_gpio_change_mode(ddc->pin_data, mode);
 | |
| 
 | |
| 	/* [anaumov] DAL2 code returns GPIO_RESULT_NON_SPECIFIC_ERROR
 | |
| 	 * in case of failures;
 | |
| 	 * set_mode() is so that, in case of failure,
 | |
| 	 * we must explicitly set original mode */
 | |
| 
 | |
| 	if (result != GPIO_RESULT_OK)
 | |
| 		goto failure;
 | |
| 
 | |
| 	result = dal_gpio_change_mode(ddc->pin_clock, mode);
 | |
| 
 | |
| 	if (result == GPIO_RESULT_OK)
 | |
| 		return result;
 | |
| 
 | |
| 	dal_gpio_change_mode(ddc->pin_clock, original_mode);
 | |
| 
 | |
| failure:
 | |
| 	dal_gpio_change_mode(ddc->pin_data, original_mode);
 | |
| 
 | |
| 	return result;
 | |
| }
 | |
| 
 | |
| enum gpio_ddc_line dal_ddc_get_line(
 | |
| 	const struct ddc *ddc)
 | |
| {
 | |
| 	return (enum gpio_ddc_line)dal_gpio_get_enum(ddc->pin_data);
 | |
| }
 | |
| 
 | |
| enum gpio_result dal_ddc_set_config(
 | |
| 	struct ddc *ddc,
 | |
| 	enum gpio_ddc_config_type config_type)
 | |
| {
 | |
| 	struct gpio_config_data config_data;
 | |
| 
 | |
| 	config_data.type = GPIO_CONFIG_TYPE_DDC;
 | |
| 
 | |
| 	config_data.config.ddc.type = config_type;
 | |
| 	config_data.config.ddc.data_en_bit_present = false;
 | |
| 	config_data.config.ddc.clock_en_bit_present = false;
 | |
| 
 | |
| 	return dal_gpio_set_config(ddc->pin_data, &config_data);
 | |
| }
 | |
| 
 | |
| void dal_ddc_close(
 | |
| 	struct ddc *ddc)
 | |
| {
 | |
| 	dal_gpio_close(ddc->pin_clock);
 | |
| 	dal_gpio_close(ddc->pin_data);
 | |
| }
 | |
| 
 |