cregit-Linux how code gets into the kernel

Release 4.16 drivers/soundwire/intel_init.c

// SPDX-License-Identifier: (GPL-2.0 OR BSD-3-Clause)
// Copyright(c) 2015-17 Intel Corporation.

/*
 * SDW Intel Init Routines
 *
 * Initializes and creates SDW devices based on ACPI and Hardware values
 */

#include <linux/acpi.h>
#include <linux/platform_device.h>
#include <linux/soundwire/sdw_intel.h>
#include "intel.h"


#define SDW_MAX_LINKS		4

#define SDW_SHIM_LCAP		0x0

#define SDW_SHIM_BASE		0x2C000

#define SDW_ALH_BASE		0x2C800

#define SDW_LINK_BASE		0x30000

#define SDW_LINK_SIZE		0x10000


struct sdw_link_data {
	
struct sdw_intel_link_res res;
	
struct platform_device *pdev;
};


struct sdw_intel_ctx {
	
int count;
	
struct sdw_link_data *links;
};


static int sdw_intel_cleanup_pdev(struct sdw_intel_ctx *ctx) { struct sdw_link_data *link = ctx->links; int i; if (!link) return 0; for (i = 0; i < ctx->count; i++) { if (link->pdev) platform_device_unregister(link->pdev); link++; } kfree(ctx->links); ctx->links = NULL; return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Vinod Koul80100.00%1100.00%
Total80100.00%1100.00%


static struct sdw_intel_ctx *sdw_intel_add_controller(struct sdw_intel_res *res) { struct platform_device_info pdevinfo; struct platform_device *pdev; struct sdw_link_data *link; struct sdw_intel_ctx *ctx; struct acpi_device *adev; int ret, i; u8 count; u32 caps; if (acpi_bus_get_device(res->handle, &adev)) return NULL; /* Found controller, find links supported */ count = 0; ret = fwnode_property_read_u8_array(acpi_fwnode_handle(adev), "mipi-sdw-master-count", &count, 1); /* Don't fail on error, continue and use hw value */ if (ret) { dev_err(&adev->dev, "Failed to read mipi-sdw-master-count: %d\n", ret); count = SDW_MAX_LINKS; } /* Check SNDWLCAP.LCOUNT */ caps = ioread32(res->mmio_base + SDW_SHIM_BASE + SDW_SHIM_LCAP); /* Check HW supported vs property value and use min of two */ count = min_t(u8, caps, count); /* Check count is within bounds */ if (count > SDW_MAX_LINKS) { dev_err(&adev->dev, "Link count %d exceeds max %d\n", count, SDW_MAX_LINKS); return NULL; } dev_dbg(&adev->dev, "Creating %d SDW Link devices\n", count); ctx = kzalloc(sizeof(*ctx), GFP_KERNEL); if (!ctx) return NULL; ctx->count = count; ctx->links = kcalloc(ctx->count, sizeof(*ctx->links), GFP_KERNEL); if (!ctx->links) goto link_err; link = ctx->links; /* Create SDW Master devices */ for (i = 0; i < count; i++) { link->res.irq = res->irq; link->res.registers = res->mmio_base + SDW_LINK_BASE + (SDW_LINK_SIZE * i); link->res.shim = res->mmio_base + SDW_SHIM_BASE; link->res.alh = res->mmio_base + SDW_ALH_BASE; memset(&pdevinfo, 0, sizeof(pdevinfo)); pdevinfo.parent = res->parent; pdevinfo.name = "int-sdw"; pdevinfo.id = i; pdevinfo.fwnode = acpi_fwnode_handle(adev); pdevinfo.data = &link->res; pdevinfo.size_data = sizeof(link->res); pdev = platform_device_register_full(&pdevinfo); if (IS_ERR(pdev)) { dev_err(&adev->dev, "platform device creation failed: %ld\n", PTR_ERR(pdev)); goto pdev_err; } link->pdev = pdev; link++; } return ctx; pdev_err: sdw_intel_cleanup_pdev(ctx); link_err: kfree(ctx); return NULL; }

Contributors

PersonTokensPropCommitsCommitProp
Vinod Koul430100.00%1100.00%
Total430100.00%1100.00%


static acpi_status sdw_intel_acpi_cb(acpi_handle handle, u32 level, void *cdata, void **return_value) { struct sdw_intel_res *res = cdata; struct acpi_device *adev; if (acpi_bus_get_device(handle, &adev)) { dev_err(&adev->dev, "Couldn't find ACPI handle\n"); return AE_NOT_FOUND; } res->handle = handle; return AE_OK; }

Contributors

PersonTokensPropCommitsCommitProp
Vinod Koul67100.00%1100.00%
Total67100.00%1100.00%

/** * sdw_intel_init() - SoundWire Intel init routine * @parent_handle: ACPI parent handle * @res: resource data * * This scans the namespace and creates SoundWire link controller devices * based on the info queried. */
void *sdw_intel_init(acpi_handle *parent_handle, struct sdw_intel_res *res) { acpi_status status; status = acpi_walk_namespace(ACPI_TYPE_DEVICE, parent_handle, 1, sdw_intel_acpi_cb, NULL, res, NULL); if (ACPI_FAILURE(status)) return NULL; return sdw_intel_add_controller(res); }

Contributors

PersonTokensPropCommitsCommitProp
Vinod Koul53100.00%1100.00%
Total53100.00%1100.00%

EXPORT_SYMBOL(sdw_intel_init); /** * sdw_intel_exit() - SoundWire Intel exit * @arg: callback context * * Delete the controller instances created and cleanup */
void sdw_intel_exit(void *arg) { struct sdw_intel_ctx *ctx = arg; sdw_intel_cleanup_pdev(ctx); kfree(ctx); }

Contributors

PersonTokensPropCommitsCommitProp
Vinod Koul26100.00%1100.00%
Total26100.00%1100.00%

EXPORT_SYMBOL(sdw_intel_exit); MODULE_LICENSE("Dual BSD/GPL"); MODULE_DESCRIPTION("Intel Soundwire Init Library");

Overall Contributors

PersonTokensPropCommitsCommitProp
Vinod Koul744100.00%1100.00%
Total744100.00%1100.00%
Information contained on this website is for historical information purposes only and does not indicate or represent copyright ownership.
Created with cregit.