Files
kernel-tenderloin-3.0/arch/arm/mach-msm/msm_reqs.c
Bryan Huntsman 3f2bc4d6eb Initial Contribution
msm-2.6.38: tag AU_LINUX_ANDROID_GINGERBREAD.02.03.04.00.142

Signed-off-by: Bryan Huntsman <bryanh@codeaurora.org>
2011-10-03 09:57:10 -07:00

126 lines
3.1 KiB
C

/* Copyright (c) 2010, Code Aurora Forum. All rights reserved.
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License version 2 and
* only version 2 as published by the Free Software Foundation.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
*/
#include <linux/kernel.h>
#include <linux/module.h>
#include <linux/completion.h>
#include <linux/err.h>
#include <mach/msm_reqs.h>
#include "npa_remote.h"
#include <linux/slab.h>
struct request {
atomic_t creation_pending;
struct npa_client *client;
char *client_name;
char *resource_name;
};
/**
* npa_res_available_cb - NPA resource-available callback.
* @u_data: Pointer to request struct.
* @t,d,n: Unused
*/
static void npa_res_available_cb(void *u_data, unsigned t, void *d, unsigned n)
{
struct request *request = u_data;
/* Make sure client is still needed. */
if (!atomic_dec_and_test(&request->creation_pending)) {
kfree(request);
return;
}
/* Create NPA 'required' client. */
request->client = npa_create_sync_client(request->resource_name,
request->client_name, NPA_CLIENT_REQUIRED);
if (IS_ERR(request->client)) {
pr_crit("npa_req: Failed to create NPA client '%s' "
"for resource '%s'. (Error %ld)\n",
request->client_name, request->resource_name,
PTR_ERR(request->client));
BUG();
}
return;
}
/**
* msm_req_add - Creates an NPA request and returns a handle. Non-blocking.
* @req_name: Name of the request
* @res_name: Name of the NPA resource the request is for
*/
void *msm_req_add(char *res_name, char *req_name)
{
struct request *request;
request = kmalloc(sizeof(*request), GFP_KERNEL);
if (!request)
return ERR_PTR(-ENOMEM);
/* Populate request data. */
request->client_name = req_name;
request->resource_name = res_name;
/* Mark client creation as pending. */
atomic_set(&request->creation_pending, 1);
/* Create NPA client when the resource becomes available. */
npa_resource_available(res_name, npa_res_available_cb, request);
return request;
}
/**
* msm_req_update - Updates an existing NPA request. May block.
* @req: Request handle
* @value: Request value
*/
int msm_req_update(void *req, s32 value)
{
struct request *request = req;
int rc = 0;
if (atomic_read(&request->creation_pending)) {
pr_err("%s: Error: No client '%s' for resource '%s'.\n",
__func__, request->client_name, request->resource_name);
return -ENXIO;
}
if (value == MSM_REQ_DEFAULT_VALUE)
npa_complete_request(request->client);
else
rc = npa_issue_required_request(request->client, value);
return rc;
}
/**
* msm_req_remove - Removes an existing NPA request. May block.
* @req: Request handle
*/
int msm_req_remove(void *req)
{
struct request *request = req;
/* Remove client if it's been created. */
if (!atomic_dec_and_test(&request->creation_pending)) {
npa_cancel_request(request->client);
npa_destroy_client(request->client);
kfree(request);
}
return 0;
}