Re: [PATCH v5] soc: qcom: Add SoC info driver

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



On 12/5/2016 11:35 PM, Bjorn Andersson wrote:
> On Wed 16 Nov 05:22 PST 2016, Imran Khan wrote:
> 
<snip>
>>  
>> diff --git a/drivers/soc/qcom/socinfo.c b/drivers/soc/qcom/socinfo.c
>> new file mode 100644
>> index 0000000..eca46df
>> --- /dev/null
>> +++ b/drivers/soc/qcom/socinfo.c
>> @@ -0,0 +1,989 @@
>> +/*
>> + * Copyright (c) 2009-2016, The Linux Foundation. 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/export.h>
>> +#include <linux/module.h>
>> +#include <linux/err.h>
>> +#include <linux/of.h>
>> +#include <linux/platform_device.h>
>> +#include <linux/sys_soc.h>
>> +#include <linux/slab.h>
>> +#include <linux/stat.h>
>> +#include <linux/string.h>
>> +#include <linux/types.h>
>> +#include <linux/random.h>
>> +#include <linux/soc/qcom/smem.h>
>> +
>> +#define PMIC_MODEL_UNKNOWN 0
>> +#define HW_PLATFORM_QRD 11
>> +#define PLATFORM_SUBTYPE_QRD_INVALID 6
>> +#define PLATFORM_SUBTYPE_INVALID 4
>> +/*
>> + * SOC version type with major number in the upper 16 bits and minor
>> + * number in the lower 16 bits.  For example:
>> + *   1.0 -> 0x00010000
>> + *   2.3 -> 0x00020003
>> + */
>> +#define SOC_VERSION_MAJOR(ver) (((ver) & 0xffff0000) >> 16)
>> +#define SOC_VERSION_MINOR(ver) ((ver) & 0x0000ffff)
>> +#define SOCINFO_FORMAT(x)  (x)
>> +#define SOCINFO_VERSION_MAJOR	SOC_VERSION_MAJOR
>> +#define SOCINFO_VERSION_MINOR	SOC_VERSION_MINOR
>> +
>> +#define SMEM_SOCINFO_BUILD_ID_LENGTH 32
>> +#define SMEM_IMAGE_VERSION_BLOCKS_COUNT 32
>> +#define SMEM_IMAGE_VERSION_SIZE 4096
>> +#define SMEM_IMAGE_VERSION_NAME_SIZE 75
>> +#define SMEM_IMAGE_VERSION_VARIANT_SIZE 20
>> +#define SMEM_IMAGE_VERSION_OEM_SIZE 32
>> +#define SMEM_IMAGE_VERSION_PARTITION_APPS 10
> 
> Please indent the values evenly, making this slightly easier to read.
> 
Done.

>> +
>> +/*
>> + * Shared memory identifiers, used to acquire handles to respective memory
>> + * region.
>> + */
>> +#define SMEM_IMAGE_VERSION_TABLE	469
>> +
>> +/*
>> + * Shared memory identifiers, used to acquire handles to respective memory
>> + * region.
>> + */
> 
> Duplicating this comment doesn't add value, please just drop this second
> copy and replace the above comment with something like "SMEM item ids"
> 
Done.
>> +#define SMEM_HW_SW_BUILD_ID		137
>> +
>> +/*
>> + * SMEM Image table indices
>> + */
<snip>
> 
> Rather than having this based on two huge macros take a look at
> DEVICE_INT_ATTR() and struct dev_ext_attribute in
> include/linux/device.h. It looks like if you just put the index in a
> struct and use container_of to reach that you can replace these with
> direct functions.
> 
> Also, it's perfectly fine to always specify a store operation and use
> 0444 vs 0644 to control it's availability. So you don't need two sets if
> you just expose the mode in your macro.
>

If I understood this correct, the main idea here is to use dev_ext_attribute
and avoid wrapper macros. Have tried to implement this suggestion in the 
subsequent patch set. Please let me know if it looks okay or can be improved
further. 
 
>> +
>> +/*
>> + * Qcom SoC types
>> + */
>> +enum qcom_cpu {
>> +	MSM_CPU_UNKNOWN = 0,
>> +	MSM_CPU_8960,
>> +	MSM_CPU_8960AB,
>> +	MSM_CPU_8064,
>> +	MSM_CPU_8974,
>> +	MSM_CPU_8974PRO_AA,
>> +	MSM_CPU_8974PRO_AB,
>> +	MSM_CPU_8974PRO_AC,
>> +	MSM_CPU_8916,
>> +	MSM_CPU_8084,
>> +	MSM_CPU_8996
>> +};
> 
> These are now "unused" (per below comment).
> 

Done. As these are no more needed.
>> +
>> +struct qcom_soc_info {
>> +	enum qcom_cpu generic_soc_type;
> 
> This enum is no longer used, seems like you can turn cpu_of_id into a
> simple char *[] array with with soc_id_string indexed by id.
> 
True. Have implemented cpu_of_id as per this suggestion.

>> +	char *soc_id_string;
>> +};
>> +
<snip>
>> +
>> +static struct smem_image_version *smem_image_version;
>> +static bool smem_img_tbl_avlbl = true;
> 
> You don't need to keep track of this separately, just test if
> smem_image_version is NULL or not (reset it to NULL if qcom_smem_get()
> fails).
> 
Done.

>> +static bool hw_subtype_valid = true;
>> +static u32 hw_subtype;
> 
> Don't keep track of this in a global variable, it's perfectly fine to
> figure it out in qcom_get_platform_subtype()
> 
Done.
>> +
>> +
>> +/* max socinfo format version supported */
>> +#define MAX_SOCINFO_FORMAT SOCINFO_FORMAT(12)
> 
> It's fine to drop SOCINFO_FORMAT and just put 12 here.
> 
Done.
>> +
>> +static struct qcom_soc_info cpu_of_id[] = {
>> +
>> +	[0] = {MSM_CPU_UNKNOWN, "Unknown CPU"},
>> +
<snip>
>> +static u32 socinfo_format;
>> +
>> +static u32 socinfo_get_id(void);
>> +
> 
> Merge socinfo_init_sysfs() and qcom_socinfo_init() into one function and
> you don't need to declare this here.
>
Done.
 
>> +/* socinfo: sysfs functions */
>> +
>> +static char *socinfo_get_id_string(void)
>> +{
>> +	return (socinfo) ? cpu_of_id[socinfo->v0_1.id].soc_id_string : NULL;
> 
> socinfo can't be NULL anymore.
> 
True. This checking is redundant indeed. In the subsequent patch set I
have removed redundant checking of socinfo being NULL or not.

>> +}
>> +
>> +static char *socinfo_get_image_version_base_address(struct device *dev)
> 
> Inline this into qcom_socinfo_init() as described below.
> 
Done.

>> +{
>> +	size_t size, size_in;
>> +	void *ptr;
>> +
>> +	ptr = qcom_smem_get(QCOM_SMEM_HOST_ANY, SMEM_IMAGE_VERSION_TABLE,
>> +			&size);
>> +	if (IS_ERR(ptr))
>> +		return ptr;
>> +
>> +	size_in = SMEM_IMAGE_VERSION_SIZE;
>> +	if (size_in != size) {
>> +		dev_err(dev, "Wrong size for smem item\n");
>> +		return ERR_PTR(-EINVAL);
>> +	}
>> +
>> +	return ptr;
>> +}
>> +
>> +
>> +static u32 socinfo_get_version(void)
>> +{
>> +	return (socinfo) ? socinfo->v0_1.version : 0;
> 
> socinfo can't be NULL, so you can just inline "socinfo->v0_1.version" in
> the one place calling this.
> 
Done.

>> +}
>> +
>> +static char *socinfo_get_build_id(void)
>> +{
>> +	return (socinfo) ? socinfo->v0_1.build_id : NULL;
> 
> socinfo can't be NULL, and this is only called from one place, so inline
> "socinfo->v0_1.build_id" there.
> 
Done.
>> +}
>> +
>> +static u32 socinfo_get_raw_version(void)
>> +{
>> +	return socinfo ?
> 
> socinfo can't be NULL.
> 
>> +		(socinfo_format >= SOCINFO_FORMAT(2) ?
> 
As mentioned above, have taken care of this.

> qcom_soc_attr_raw_version will not be created unless socinfo_format >=
> 2, so this can't be called if socinfo_format < 2, so you don't need this
> extra check.
> 
>> +			socinfo->v0_2.raw_version : 0)
>> +		: 0;
> 
> And as this is then simplified to "return socinfo->v0_2.raw_version;" I
> think you should just inline it in the one place calling this function.
> 
Done. Have removed redundant checking of socinfo format version from 
this and other functions and also have made these functions as much inline
as possible.

>> +}
>> +
>> +static u32 socinfo_get_platform_type(void)
>> +{
>> +	return socinfo ?
>> +		(socinfo_format >= SOCINFO_FORMAT(3) ?
>> +			socinfo->v0_3.hw_platform : 0)
>> +		: 0;
>> +}
<snip>

>> +
>> +static ssize_t
>> +qcom_get_raw_version(struct device *dev,
>> +		     struct device_attribute *attr,
>> +		     char *buf)
> 
> This is the "show" function of an attr, so please drop the "get" and add
> "_show".
> 
Done. Have renamed attr show and store functions properly.
>> +{
>> +	return sprintf(buf, "%u\n",
>> +		socinfo_get_raw_version());
> 
> And please inline as much as you can of these, there's little point in
> having a helper function to wrap a single line accessing the element in
> the socinfo struct.
> 

Done. Have tried this but please let me know if you see any further scope
in this regard.

>> +}
>> +
>> +static ssize_t
>> +qcom_get_build_id(struct device *dev,
>> +		   struct device_attribute *attr,
>> +		   char *buf)
>> +{
>> +	return scnprintf(buf, SMEM_SOCINFO_BUILD_ID_LENGTH, "%s\n",
>> +				socinfo_get_build_id());
>> +}
>> +
<snip>
>> +
>> +static ssize_t
>> +qcom_get_pmic_die_revision(struct device *dev,
>> +			       struct device_attribute *attr,
>> +			       char *buf)
>> +{
>> +	return sprintf(buf, "%u\n", socinfo_get_pmic_die_revision());
>> +}
>> +
>> +static ssize_t
>> +qcom_get_image_version(int index, char *buf)
>> +{
> 
> Just inline these into the version show() and store() functions.  And
> whenever you copy to the sysfs buffer, it's of size PAGE_SIZE.
> 
Done.
>> +	return scnprintf(buf, SMEM_IMAGE_VERSION_NAME_SIZE, "%s\n",
>> +			smem_image_version[index].name);
>> +}
>> +
<snip>
>> +static ssize_t
>> +qcom_set_image_crm_version(int index, const char *buf)
>> +{
>> +	return strlcpy(smem_image_version[index].oem, buf,
>> +			SMEM_IMAGE_VERSION_OEM_SIZE);
>> +}
>> +
>> +static struct device_attribute qcom_soc_attr_raw_version =
>> +	__ATTR(raw_version, 0444, qcom_get_raw_version,  NULL);
> 
> Rather than writing all these out here you can put them in an array of:
> 
> struct qcom_socinfo_attr {
> 	struct device_attribute attr;
> 	int min_version;
> };
> 
> #define QCOM_SOCINFO_ATTR(_name, _show, _min_version) \
> 	{ __ATTR(_name, 0444, _show, NULL), _min_version) }
> 
> static const struct qcom_socinfo_attr qcom_socinfo_attrs[] = {
> 	QCOM_SOCINFO_ATTR(raw_version, qcom_get_raw_version, 2),
> };
> 
> And then socinfo_populate_sysfs_files() will be a matter of looping over
> this array and comparing min_version and registering the attr.
>
Done. This makes the implementation really neat and concise. Thanks for
this suggestion.
 
>> +
>> +static struct device_attribute qcom_soc_attr_vendor =
>> +	__ATTR(vendor, 0444, qcom_get_vendor,  NULL);
>> +
>> +static struct device_attribute qcom_soc_attr_build_id =
>> +	__ATTR(build_id, 0444, qcom_get_build_id, NULL);
<snip>
>> +	/*
>> +	 * Expose SMEM_IMAGE_TABLE to sysfs only when we have IMAGE_TABLE
>> +	 * available in SMEM. As IMAGE_TABLE and SOCINFO are two separate
>> +	 * items within SMEM, we expose the remaining soc information(i.e
>> +	 * only the SOCINFO item available in SMEM) to sysfs even in the
>> +	 * absence of an IMAGE_TABLE.
>> +	 */
>> +	if (smem_img_tbl_avlbl) {
>> +		for (img_idx = 0; img_idx < SMEM_IMAGE_VERSION_BLOCKS_COUNT;
>> +				img_idx++) {
>> +			if (smem_image_table[img_idx]) {
>> +				err = sysfs_create_group(&qcom_soc_device->kobj,
>> +					smem_image_table[img_idx]);
>> +				if (err)
>> +					break;
> 
> Should we just ignore this error?
> 
Yeah. I think we can and should ignore this error. Failing to create sysfs entry
for one item in image table should not deprive other SMEM image items.
>> +			}
>> +		}
>> +	}
>> +	err = device_create_file(qcom_soc_device, &qcom_soc_attr_vendor);
>> +	if (!err) {
> 
> Flip this conditional around and handle the error here, it saves you one
> indentation level for the rest of this function.
> 
Done. No more needed in new implementation.

>> +		switch (socinfo_format) {
>> +		case SOCINFO_FORMAT(12):
>> +			err = device_create_file(qcom_soc_device,
>> +					&qcom_soc_attr_chip_family);
>> +			if (!err)
>> +				err = device_create_file(qcom_soc_device,
>> +					&qcom_soc_attr_raw_device_family);
>> +			if (!err)
>> +				err = device_create_file(qcom_soc_device,
>> +					&qcom_soc_attr_raw_device_number);
<snip>
>> +			return 0;
>> +		default:
>> +			dev_err(qcom_soc_device, "Unknown socinfo format: v0.%u\n",
>> +					SOCINFO_VERSION_MINOR(socinfo_format));
>> +			return -EINVAL;
>> +		}
>> +	} else {
>> +		dev_err(qcom_soc_device, "Could not create sysfs entry\n");
>> +		return err;
>> +	}
>> +}
>> +
>> +static int socinfo_init_sysfs(struct device *dev)
> 
> This device * is specifically the smem device, so if you name it
> "smem_dev"...
> 
Done.
>> +{
>> +	struct device *qcom_soc_device;
> 
> ...then you can name this "dev" to shorten things.
>
Done.
 
>> +	struct soc_device *soc_dev;
>> +	struct soc_device_attribute *soc_dev_attr;
> 
> "attr" would be long enough.
>
Done.
 
>> +	u32 soc_version;
>> +
>> +	soc_dev_attr = kzalloc(sizeof(*soc_dev_attr), GFP_KERNEL);
>> +	if (!soc_dev_attr)
>> +		return -ENOMEM;
>> +
>> +	soc_version = socinfo_get_version();
>> +
>> +	soc_dev_attr->soc_id = kasprintf(GFP_KERNEL, "%d", socinfo_get_id());
>> +	soc_dev_attr->family = "Snapdragon";
>> +	soc_dev_attr->machine = socinfo_get_id_string();
> 
> Single use of socinfo_get_id_string() so you should just go:
> 
> unsigned int id = socinfo->v0_1.id;
> 
> and then use that for your soc_id print and use soc_id_string[id] here.
> 
Done.
>> +	soc_dev_attr->revision = kasprintf(GFP_KERNEL, "%u.%u",
>> +				SOC_VERSION_MAJOR(soc_version),
>> +				SOC_VERSION_MINOR(soc_version));
>> +
>> +	soc_dev = soc_device_register(soc_dev_attr);
>> +	if (IS_ERR(soc_dev)) {
>> +		kfree(soc_dev_attr);
>> +		return PTR_ERR(soc_dev);
>> +	}
>> +
>> +	qcom_soc_device = soc_device_to_device(soc_dev);
>> +	socinfo_populate_sysfs_files(qcom_soc_device);
>> +	return 0;
>> +}
>> +
>> +static u32 socinfo_get_id(void)
>> +{
>> +	return (socinfo) ? socinfo->v0_1.id : 0;
> 
> As we no longer expose the socinfo functions to external callers this
> will only ever be called if socinfo_init() succeeded, so no need to
> check for socinfo being NULL.
> 
Done.
>> +}
>> +
>> +static int socinfo_select_format(struct device *dev)
>> +{
>> +	u32 f_maj = SOCINFO_VERSION_MAJOR(socinfo->v0_1.format);
>> +	u32 f_min = SOCINFO_VERSION_MINOR(socinfo->v0_1.format);
>> +
>> +	if (f_maj != 0) {
>> +		dev_err(dev, "Unsupported format v%u.%u.\n",
>> +			f_maj, f_min);
>> +		return -EINVAL;
>> +	}
>> +
>> +	if (socinfo->v0_1.format > MAX_SOCINFO_FORMAT) {
>> +		dev_warn(dev, "Unsupported format v%u.%u. Falling back to v%u.%u.\n",
>> +			f_maj, f_min, SOCINFO_VERSION_MAJOR(MAX_SOCINFO_FORMAT),
>> +			SOCINFO_VERSION_MINOR(MAX_SOCINFO_FORMAT));
>> +		socinfo_format = MAX_SOCINFO_FORMAT;
> 
> Don't fall back, just print an error and fail.
> 
Done.
>> +	} else {
>> +		socinfo_format = socinfo->v0_1.format;
>> +	}
>> +	return 0;
>> +}
>> +
>> +int qcom_socinfo_init(struct platform_device *pdev)
> 
> As smem doesn't care if we're failing here you can just have the return
> type void.
> 
Done.
>> +{
>> +	size_t size;
>> +
>> +	socinfo = qcom_smem_get(QCOM_SMEM_HOST_ANY, SMEM_HW_SW_BUILD_ID,
>> +			&size);
>> +	if (IS_ERR(socinfo) || (socinfo_select_format(&pdev->dev) < 0)) {
>> +		dev_warn(&pdev->dev,
>> +			"Coudn't find soc information; Unable to setup socinfo.\n");
>> +		return -ENOMEM;
> 
> "Couldn't find socinfo" and return -ENOENT if IS_ERR(socinfo).
> 
Done
>> +	}
> 
> And then just move the rest of socinfo_select_format() here:
> 
> 	if (socinfo->v0.1.format > MAX_SOCINFO_FORMAT) {
> 		dev_err(&pdev->dev, "Invalid socinfo version %d.%d\n",
> 			MAJOR(), MINOR());
> 		return -EINVAL;
> 	}
> 
Done
>> +
>> +	if (!socinfo_get_id()) {
>> +		dev_err(&pdev->dev, "socinfo: Unknown SoC ID!\n");
>> +		return -EINVAL;
>> +	}
>> +
>> +	WARN(socinfo_get_id() >= ARRAY_SIZE(cpu_of_id),
>> +		"New IDs added! ID => CPU mapping needs an update.\n");
>> +
>> +	smem_image_version = (struct smem_image_version *)
>> +			socinfo_get_image_version_base_address(&pdev->dev);
> 
> You could make socinfo_get_image_version_base_address() return void *
> and you don't have to do the type cast. But you can make this more
> compact by just inlining the qcom_smem_get() here directly.
> 
> 	smem_image_version = qcom_smem_get(QCOM_SMEM_HOST_ANY,
> 					   SMEM_IMAGE_VERSION_TABLE,
> 					   &size);
> 	if (IS_ERR(smem_image_version) || size != SMEM_IMAGE_VERSION_SIZE) {
> 		...
> 	}
> 

Done
>> +	if (IS_ERR(smem_image_version)) {
>> +		dev_warn(&pdev->dev, "Unable to get address for image version table.\n");
> 
> "Image version table missing" should keep you < 80 characters.
> 
Done.
>> +		smem_img_tbl_avlbl = false;
>> +	}
>> +
>> +	hw_subtype = socinfo_get_platform_subtype();
>> +	if (socinfo_get_platform_type() == HW_PLATFORM_QRD) {
>> +		if (hw_subtype >= PLATFORM_SUBTYPE_QRD_INVALID) {
>> +			dev_err(&pdev->dev, "Invalid hardware platform sub type for qrd found\n");
>> +			hw_subtype_valid = false;
>> +		}
>> +	} else {
>> +		if (hw_subtype >= PLATFORM_SUBTYPE_INVALID) {
>> +			dev_err(&pdev->dev, "Invalid hardware platform subtype\n");
>> +			hw_subtype_valid = false;
>> +		}
>> +	}
> 
> This will only matter if format >= 6 and hw_subtype is outside the array
> of strings, I think you should just be optimistic about this not being
> an issue and then in all cases where you dereference a lookup table
> check that you're within bounds, or fail with -EINVAL; to userspace.
> 
True. Have taken care of this. No more tracking this with a global variable.
>> +
>> +	socinfo_init_sysfs(&pdev->dev);
> 
> socinfo_init_sysfs() is just the second half of this function, so inline
> it here.
> 
Done. now this is inline.
>> +
>> +	/* Feed the soc specific unique data into entropy pool */
>> +	add_device_randomness(socinfo, size);
>> +
>> +	return 0;
>> +}
>> +EXPORT_SYMBOL(qcom_socinfo_init);
>> +
> 
> No empty lines at the end of the file, please.
> 
Done.
> Regards,
> Bjorn
> 
Thanks Bjorn for review comments.
Regards,
Imran

-- 
QUALCOMM INDIA, on behalf of Qualcomm Innovation Center, Inc. is a\nmember of the Code Aurora Forum, hosted by The Linux Foundation
--
To unsubscribe from this list: send the line "unsubscribe linux-arm-msm" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at  http://vger.kernel.org/majordomo-info.html



[Index of Archives]     [Linux ARM Kernel]     [Linux ARM]     [Linux Omap]     [Fedora ARM]     [Linux for Sparc]     [IETF Annouce]     [Security]     [Bugtraq]     [Linux MIPS]     [ECOS]     [Asterisk Internet PBX]     [Linux API]

  Powered by Linux