*/
 
 #include "dm_services.h"
+
 #include "include/gpio_types.h"
 #include "../hw_factory.h"
 
+#include "../hw_gpio.h"
+#include "../hw_ddc.h"
+#include "../hw_hpd.h"
+#include "../hw_generic.h"
+
 #include "hw_factory_dce110.h"
 
 #include "dce/dce_11_0_d.h"
 }
 
 static const struct hw_factory_funcs funcs = {
-       .create_ddc_data = dal_hw_ddc_create,
-       .create_ddc_clock = dal_hw_ddc_create,
-       .create_generic = NULL,
-       .create_hpd = dal_hw_hpd_create,
-       .create_sync = NULL,
-       .create_gsl = NULL,
+       .init_ddc_data = dal_hw_ddc_init,
+       .init_generic = NULL,
+       .init_hpd = dal_hw_hpd_init,
+       .get_ddc_pin = dal_hw_ddc_get_pin,
+       .get_hpd_pin = dal_hw_hpd_get_pin,
+       .get_generic_pin = NULL,
        .define_hpd_registers = define_hpd_registers,
        .define_ddc_registers = define_ddc_registers
 };
 
 #include "include/gpio_types.h"
 #include "../hw_factory.h"
 
-
 #include "../hw_gpio.h"
 #include "../hw_ddc.h"
 #include "../hw_hpd.h"
+#include "../hw_generic.h"
 
 #include "hw_factory_dce120.h"
 
 
 /* fucntion table */
 static const struct hw_factory_funcs funcs = {
-       .create_ddc_data = dal_hw_ddc_create,
-       .create_ddc_clock = dal_hw_ddc_create,
-       .create_generic = NULL,
-       .create_hpd = dal_hw_hpd_create,
-       .create_sync = NULL,
-       .create_gsl = NULL,
+       .init_ddc_data = dal_hw_ddc_init,
+       .init_generic = NULL,
+       .init_hpd = dal_hw_hpd_init,
+       .get_ddc_pin = dal_hw_ddc_get_pin,
+       .get_hpd_pin = dal_hw_hpd_get_pin,
+       .get_generic_pin = NULL,
        .define_hpd_registers = define_hpd_registers,
        .define_ddc_registers = define_ddc_registers
 };
 
 #include "../hw_gpio.h"
 #include "../hw_ddc.h"
 #include "../hw_hpd.h"
+#include "../hw_generic.h"
 
 #include "dce/dce_8_0_d.h"
 #include "dce/dce_8_0_sh_mask.h"
 
+
 #define REG(reg_name)\
                mm ## reg_name
 
 }
 
 static const struct hw_factory_funcs funcs = {
-       .create_ddc_data = dal_hw_ddc_create,
-       .create_ddc_clock = dal_hw_ddc_create,
-       .create_generic = NULL,
-       .create_hpd = dal_hw_hpd_create,
-       .create_sync = NULL,
-       .create_gsl = NULL,
+       .init_ddc_data = dal_hw_ddc_init,
+       .init_generic = NULL,
+       .init_hpd = dal_hw_hpd_init,
+       .get_ddc_pin = dal_hw_ddc_get_pin,
+       .get_hpd_pin = dal_hw_hpd_get_pin,
+       .get_generic_pin = NULL,
        .define_hpd_registers = define_hpd_registers,
        .define_ddc_registers = define_ddc_registers
 };
 
 
 /* fucntion table */
 static const struct hw_factory_funcs funcs = {
-       .create_ddc_data = dal_hw_ddc_create,
-       .create_ddc_clock = dal_hw_ddc_create,
-       .create_generic = dal_hw_generic_create,
-       .create_hpd = dal_hw_hpd_create,
-       .create_sync = NULL,
-       .create_gsl = NULL,
+       .init_ddc_data = dal_hw_ddc_init,
+       .init_generic = dal_hw_generic_init,
+       .init_hpd = dal_hw_hpd_init,
+       .get_ddc_pin = dal_hw_ddc_get_pin,
+       .get_hpd_pin = dal_hw_hpd_get_pin,
+       .get_generic_pin = dal_hw_generic_get_pin,
        .define_hpd_registers = define_hpd_registers,
        .define_ddc_registers = define_ddc_registers,
        .define_generic_registers = define_generic_registers
 
 
 /* fucntion table */
 static const struct hw_factory_funcs funcs = {
-       .create_ddc_data = dal_hw_ddc_create,
-       .create_ddc_clock = dal_hw_ddc_create,
-       .create_generic = dal_hw_generic_create,
-       .create_hpd = dal_hw_hpd_create,
-       .create_sync = NULL,
-       .create_gsl = NULL,
+       .init_ddc_data = dal_hw_ddc_init,
+       .init_generic = dal_hw_generic_init,
+       .init_hpd = dal_hw_hpd_init,
+       .get_ddc_pin = dal_hw_ddc_get_pin,
+       .get_hpd_pin = dal_hw_hpd_get_pin,
+       .get_generic_pin = dal_hw_generic_get_pin,
        .define_hpd_registers = define_hpd_registers,
        .define_ddc_registers = define_ddc_registers,
        .define_generic_registers = define_generic_registers,
 
 
 /* function table */
 static const struct hw_factory_funcs funcs = {
-       .create_ddc_data = NULL,
-       .create_ddc_clock = NULL,
-       .create_generic = NULL,
-       .create_hpd = NULL,
-       .create_sync = NULL,
-       .create_gsl = NULL,
+       .init_ddc_data = NULL,
+       .init_generic = NULL,
+       .init_hpd = NULL,
 };
 
 void dal_hw_factory_diag_fpga_init(struct hw_factory *factory)
 
                return GPIO_RESULT_ALREADY_OPENED;
        }
 
+       // No action if allocation failed during gpio construct
+       if (!gpio->hw_container.ddc) {
+               ASSERT_CRITICAL(false);
+               return GPIO_RESULT_NON_SPECIFIC_ERROR;
+       }
        gpio->mode = mode;
 
-       return dal_gpio_service_open(
-               gpio->service, gpio->id, gpio->en, mode, &gpio->pin);
+       return dal_gpio_service_open(gpio);
 }
 
 enum gpio_result dal_gpio_get_value(
        return gpio->output_state;
 }
 
+struct hw_ddc *dal_gpio_get_ddc(struct gpio *gpio)
+{
+       return gpio->hw_container.ddc;
+}
+
+struct hw_hpd *dal_gpio_get_hpd(struct gpio *gpio)
+{
+       return gpio->hw_container.hpd;
+}
+
+struct hw_generic *dal_gpio_get_generic(struct gpio *gpio)
+{
+       return gpio->hw_container.generic;
+}
+
 void dal_gpio_close(
        struct gpio *gpio)
 {
        gpio->mode = GPIO_MODE_UNKNOWN;
        gpio->output_state = output_state;
 
+       //initialize hw_container union based on id
+       switch (gpio->id) {
+       case GPIO_ID_DDC_DATA:
+               gpio->service->factory.funcs->init_ddc_data(&gpio->hw_container.ddc, service->ctx, id, en);
+               break;
+       case GPIO_ID_DDC_CLOCK:
+               gpio->service->factory.funcs->init_ddc_data(&gpio->hw_container.ddc, service->ctx, id, en);
+               break;
+       case GPIO_ID_GENERIC:
+               gpio->service->factory.funcs->init_generic(&gpio->hw_container.generic, service->ctx, id, en);
+               break;
+       case GPIO_ID_HPD:
+               gpio->service->factory.funcs->init_hpd(&gpio->hw_container.hpd, service->ctx, id, en);
+               break;
+       // TODO: currently gpio for sync and gsl does not get created, might need it later
+       case GPIO_ID_SYNC:
+               break;
+       case GPIO_ID_GSL:
+               break;
+       default:
+               ASSERT_CRITICAL(false);
+               gpio->pin = NULL;
+       }
+
        return gpio;
 }
 
 
        dal_gpio_close(*gpio);
 
+       switch ((*gpio)->id) {
+       case GPIO_ID_DDC_DATA:
+               kfree((*gpio)->hw_container.ddc);
+               (*gpio)->hw_container.ddc = NULL;
+               break;
+       case GPIO_ID_DDC_CLOCK:
+               //TODO: might want to change it to init_ddc_clock
+               kfree((*gpio)->hw_container.ddc);
+               (*gpio)->hw_container.ddc = NULL;
+               break;
+       case GPIO_ID_GENERIC:
+               kfree((*gpio)->hw_container.generic);
+               (*gpio)->hw_container.generic = NULL;
+               break;
+       case GPIO_ID_HPD:
+               kfree((*gpio)->hw_container.hpd);
+               (*gpio)->hw_container.hpd = NULL;
+               break;
+       // TODO: currently gpio for sync and gsl does not get created, might need it later
+       case GPIO_ID_SYNC:
+               break;
+       case GPIO_ID_GSL:
+               break;
+       default:
+               break;
+       }
+
        kfree(*gpio);
 
        *gpio = NULL;
 
 }
 
 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 gpio *gpio)
 {
-       struct hw_gpio_pin *pin;
+       struct gpio_service *service = gpio->service;
+       enum gpio_id id = gpio->id;
+       uint32_t en = gpio->en;
+       enum gpio_mode mode = gpio->mode;
+
+       struct hw_gpio_pin **pin = &gpio->pin;
+
 
        if (!service->busyness[id]) {
                ASSERT_CRITICAL(false);
 
        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);
+               *pin = service->factory.funcs->get_ddc_pin(gpio);
+               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);
+               *pin = service->factory.funcs->get_ddc_pin(gpio);
+               service->factory.funcs->define_ddc_registers(*pin, en);
        break;
        case GPIO_ID_GENERIC:
-               pin = service->factory.funcs->create_generic(
-                       service->ctx, id, en);
-               service->factory.funcs->define_generic_registers(pin, en);
+               *pin = service->factory.funcs->get_generic_pin(gpio);
+               service->factory.funcs->define_generic_registers(*pin, en);
        break;
        case GPIO_ID_HPD:
-               pin = service->factory.funcs->create_hpd(
-                       service->ctx, id, en);
-               service->factory.funcs->define_hpd_registers(pin, en);
+               *pin = service->factory.funcs->get_hpd_pin(gpio);
+               service->factory.funcs->define_hpd_registers(*pin, en);
        break;
+
+       //TODO: gsl and sync support? create_sync and create_gsl are NULL
        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) {
+       if (!*pin) {
                ASSERT_CRITICAL(false);
                return GPIO_RESULT_NON_SPECIFIC_ERROR;
        }
 
-       if (!pin->funcs->open(pin, mode)) {
+       if (!(*pin)->funcs->open(*pin, mode)) {
                ASSERT_CRITICAL(false);
-               dal_gpio_service_close(service, &pin);
+               dal_gpio_service_close(service, pin);
                return GPIO_RESULT_OPEN_FAILED;
        }
 
        set_pin_busy(service, id, en);
-       *ptr = pin;
        return GPIO_RESULT_OK;
 }
 
 
                pin->funcs->close(pin);
 
-               pin->funcs->destroy(ptr);
+               *ptr = NULL;
        }
 }
 
-
 enum dc_irq_source dal_irq_get_source(
        const struct gpio *irq)
 {
 
 };
 
 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 gpio *gpio);
 
 void dal_gpio_service_close(
        struct gpio_service *service,
 
 
 #include "dm_services.h"
 
+#include "include/gpio_interface.h"
 #include "include/gpio_types.h"
 #include "hw_gpio.h"
 #include "hw_ddc.h"
 #define REG(reg)\
        (ddc->regs->reg)
 
+struct gpio;
+
 static void destruct(
        struct hw_ddc *pin)
 {
        ddc->base.base.funcs = &funcs;
 }
 
-struct hw_gpio_pin *dal_hw_ddc_create(
+void dal_hw_ddc_init(
+       struct hw_ddc **hw_ddc,
        struct dc_context *ctx,
        enum gpio_id id,
        uint32_t en)
 {
-       struct hw_ddc *pin;
-
        if ((en < GPIO_DDC_LINE_MIN) || (en > GPIO_DDC_LINE_MAX)) {
                ASSERT_CRITICAL(false);
-               return NULL;
+               *hw_ddc = NULL;
        }
 
-       pin = kzalloc(sizeof(struct hw_ddc), GFP_KERNEL);
-       if (!pin) {
+       *hw_ddc = kzalloc(sizeof(struct hw_ddc), GFP_KERNEL);
+       if (!*hw_ddc) {
                ASSERT_CRITICAL(false);
-               return NULL;
+               return;
        }
 
-       construct(pin, id, en, ctx);
-       return &pin->base.base;
+       construct(*hw_ddc, id, en, ctx);
+}
+
+struct hw_gpio_pin *dal_hw_ddc_get_pin(struct gpio *gpio)
+{
+       struct hw_ddc *hw_ddc = dal_gpio_get_ddc(gpio);
+
+       return &hw_ddc->base.base;
 }
 
 #define HW_DDC_FROM_BASE(hw_gpio) \
        container_of((HW_GPIO_FROM_BASE(hw_gpio)), struct hw_ddc, base)
 
-struct hw_gpio_pin *dal_hw_ddc_create(
+void dal_hw_ddc_init(
+       struct hw_ddc **hw_ddc,
        struct dc_context *ctx,
        enum gpio_id id,
        uint32_t en);
 
+struct hw_gpio_pin *dal_hw_ddc_get_pin(struct gpio *gpio);
+
 #endif
 
 
 struct hw_gpio_pin;
 struct hw_hpd;
+struct hw_ddc;
+struct hw_generic;
+struct gpio;
 
 struct hw_factory {
        uint32_t number_of_pins[GPIO_ID_COUNT];
 
        const struct hw_factory_funcs {
-               struct hw_gpio_pin *(*create_ddc_data)(
-                       struct dc_context *ctx,
-                       enum gpio_id id,
-                       uint32_t en);
-               struct hw_gpio_pin *(*create_ddc_clock)(
-                       struct dc_context *ctx,
-                       enum gpio_id id,
-                       uint32_t en);
-               struct hw_gpio_pin *(*create_generic)(
-                       struct dc_context *ctx,
-                       enum gpio_id id,
-                       uint32_t en);
-               struct hw_gpio_pin *(*create_hpd)(
-                       struct dc_context *ctx,
-                       enum gpio_id id,
-                       uint32_t en);
-               struct hw_gpio_pin *(*create_sync)(
-                       struct dc_context *ctx,
-                       enum gpio_id id,
-                       uint32_t en);
-               struct hw_gpio_pin *(*create_gsl)(
-                       struct dc_context *ctx,
-                       enum gpio_id id,
-                       uint32_t en);
+               void (*init_ddc_data)(
+                               struct hw_ddc **hw_ddc,
+                               struct dc_context *ctx,
+                               enum gpio_id id,
+                               uint32_t en);
+               void (*init_generic)(
+                               struct hw_generic **hw_generic,
+                               struct dc_context *ctx,
+                               enum gpio_id id,
+                               uint32_t en);
+               void (*init_hpd)(
+                               struct hw_hpd **hw_hpd,
+                               struct dc_context *ctx,
+                               enum gpio_id id,
+                               uint32_t en);
+               struct hw_gpio_pin *(*get_hpd_pin)(
+                               struct gpio *gpio);
+               struct hw_gpio_pin *(*get_ddc_pin)(
+                               struct gpio *gpio);
+               struct hw_gpio_pin *(*get_generic_pin)(
+                               struct gpio *gpio);
                void (*define_hpd_registers)(
                                struct hw_gpio_pin *pin,
                                uint32_t en);
 
 
 #include "dm_services.h"
 
+#include "include/gpio_interface.h"
 #include "include/gpio_types.h"
 #include "hw_gpio.h"
 #include "hw_generic.h"
 #define REG(reg)\
        (generic->regs->reg)
 
+struct gpio;
+
 static void dal_hw_generic_construct(
        struct hw_generic *pin,
        enum gpio_id id,
        generic->base.base.funcs = &funcs;
 }
 
-struct hw_gpio_pin *dal_hw_generic_create(
+void dal_hw_generic_init(
+       struct hw_generic **hw_generic,
        struct dc_context *ctx,
        enum gpio_id id,
        uint32_t en)
 {
-       struct hw_generic *generic;
-
-       if (id != GPIO_ID_GENERIC) {
+       if ((en < GPIO_DDC_LINE_MIN) || (en > GPIO_DDC_LINE_MAX)) {
                ASSERT_CRITICAL(false);
-               return NULL;
+               *hw_generic = NULL;
        }
 
-       if ((en < GPIO_GENERIC_MIN) || (en > GPIO_GENERIC_MAX)) {
+       *hw_generic = kzalloc(sizeof(struct hw_generic), GFP_KERNEL);
+       if (!*hw_generic) {
                ASSERT_CRITICAL(false);
-               return NULL;
+               return;
        }
 
-       generic = kzalloc(sizeof(struct hw_generic), GFP_KERNEL);
-       if (!generic) {
-               ASSERT_CRITICAL(false);
-               return NULL;
-       }
+       construct(*hw_generic, id, en, ctx);
+}
+
+
+struct hw_gpio_pin *dal_hw_generic_get_pin(struct gpio *gpio)
+{
+       struct hw_generic *hw_generic = dal_gpio_get_generic(gpio);
 
-       construct(generic, id, en, ctx);
-       return &generic->base.base;
+       return &hw_generic->base.base;
 }
 
 #define __DAL_HW_generic_H__
 
 #include "generic_regs.h"
+#include "hw_gpio.h"
 
 struct hw_generic {
        struct hw_gpio base;
 #define HW_GENERIC_FROM_BASE(hw_gpio) \
        container_of((HW_GPIO_FROM_BASE(hw_gpio)), struct hw_generic, base)
 
-struct hw_gpio_pin *dal_hw_generic_create(
+void dal_hw_generic_init(
+       struct hw_generic **hw_generic,
        struct dc_context *ctx,
        enum gpio_id id,
        uint32_t en);
 
+struct hw_gpio_pin *dal_hw_generic_get_pin(struct gpio *gpio);
+
 #endif
 
 
 #include "dm_services.h"
 
+#include "include/gpio_interface.h"
 #include "include/gpio_types.h"
 #include "hw_gpio.h"
 #include "hw_hpd.h"
 #define REG(reg)\
        (hpd->regs->reg)
 
+struct gpio;
+
 static void dal_hw_hpd_construct(
        struct hw_hpd *pin,
        enum gpio_id id,
        hpd->base.base.funcs = &funcs;
 }
 
-struct hw_gpio_pin *dal_hw_hpd_create(
+void dal_hw_hpd_init(
+       struct hw_hpd **hw_hpd,
        struct dc_context *ctx,
        enum gpio_id id,
        uint32_t en)
 {
-       struct hw_hpd *hpd;
-
-       if (id != GPIO_ID_HPD) {
+       if ((en < GPIO_DDC_LINE_MIN) || (en > GPIO_DDC_LINE_MAX)) {
                ASSERT_CRITICAL(false);
-               return NULL;
+               *hw_hpd = NULL;
        }
 
-       if ((en < GPIO_HPD_MIN) || (en > GPIO_HPD_MAX)) {
+       *hw_hpd = kzalloc(sizeof(struct hw_hpd), GFP_KERNEL);
+       if (!*hw_hpd) {
                ASSERT_CRITICAL(false);
-               return NULL;
+               return;
        }
 
-       hpd = kzalloc(sizeof(struct hw_hpd), GFP_KERNEL);
-       if (!hpd) {
-               ASSERT_CRITICAL(false);
-               return NULL;
-       }
+       construct(*hw_hpd, id, en, ctx);
+}
+
+struct hw_gpio_pin *dal_hw_hpd_get_pin(struct gpio *gpio)
+{
+       struct hw_hpd *hw_hpd = dal_gpio_get_hpd(gpio);
 
-       construct(hpd, id, en, ctx);
-       return &hpd->base.base;
+       return &hw_hpd->base.base;
 }
 
 #define HW_HPD_FROM_BASE(hw_gpio) \
        container_of((HW_GPIO_FROM_BASE(hw_gpio)), struct hw_hpd, base)
 
-struct hw_gpio_pin *dal_hw_hpd_create(
+void dal_hw_hpd_init(
+       struct hw_hpd **hw_hpd,
        struct dc_context *ctx,
        enum gpio_id id,
        uint32_t en);
 
+struct hw_gpio_pin *dal_hw_hpd_get_pin(struct gpio *gpio);
+
 #endif
 
 
 #include "gpio_types.h"
 
+
+union gpio_hw_container {
+       struct hw_ddc *ddc;
+       struct hw_generic *generic;
+       struct hw_hpd *hpd;
+};
+
 struct gpio {
        struct gpio_service *service;
        struct hw_gpio_pin *pin;
        enum gpio_id id;
        uint32_t en;
+
+       union gpio_hw_container hw_container;
        enum gpio_mode mode;
+
        /* when GPIO comes from VBIOS, it has defined output state */
        enum gpio_pin_output_state output_state;
 };
 
 enum gpio_pin_output_state dal_gpio_get_output_state(
        const struct gpio *gpio);
 
+struct hw_ddc *dal_gpio_get_ddc(struct gpio *gpio);
+
+struct hw_hpd *dal_gpio_get_hpd(struct gpio *gpio);
+
+struct hw_generic *dal_gpio_get_generic(struct gpio *gpio);
+
 /* Close the handle */
 void dal_gpio_close(
        struct gpio *gpio);
 
+
+
+
 #endif