diff mbox

[2/3] of/numa: fix a memory@ dt node can only contains one memory block

Message ID 1464230639-9852-2-git-send-email-thunder.leizhen@huawei.com
State New
Headers show

Commit Message

Leizhen (ThunderTown) May 26, 2016, 2:43 a.m. UTC
For a normal memory@ devicetree node, its reg property can contains more
memory blocks.

Because we don't known how many memory blocks maybe contained, so we try
from index=0, increase 1 until error returned(the end).

Signed-off-by: Zhen Lei <thunder.leizhen@huawei.com>

---
 drivers/of/of_numa.c | 30 ++++++++++++++++++++----------
 1 file changed, 20 insertions(+), 10 deletions(-)

--
2.5.0


--
To unsubscribe from this list: send the line "unsubscribe devicetree" in
the body of a message to majordomo@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html

Comments

Rob Herring (Arm) May 26, 2016, 1:13 p.m. UTC | #1
On Thu, May 26, 2016 at 10:43:58AM +0800, Zhen Lei wrote:
> For a normal memory@ devicetree node, its reg property can contains more

> memory blocks.

> 

> Because we don't known how many memory blocks maybe contained, so we try

> from index=0, increase 1 until error returned(the end).

> 

> Signed-off-by: Zhen Lei <thunder.leizhen@huawei.com>

> ---

>  drivers/of/of_numa.c | 30 ++++++++++++++++++++----------

>  1 file changed, 20 insertions(+), 10 deletions(-)

> 

> diff --git a/drivers/of/of_numa.c b/drivers/of/of_numa.c

> index 21d831f..2c5f249 100644

> --- a/drivers/of/of_numa.c

> +++ b/drivers/of/of_numa.c

> @@ -63,7 +63,7 @@ static int __init of_numa_parse_memory_nodes(void)

>  	struct device_node *np = NULL;

>  	struct resource rsrc;

>  	u32 nid;

> -	int r = 0;

> +	int i, r = 0;

> 

>  	for (;;) {

>  		np = of_find_node_by_type(np, "memory");

> @@ -82,17 +82,27 @@ static int __init of_numa_parse_memory_nodes(void)

>  			/* some other error */

>  			break;

> 

> -		r = of_address_to_resource(np, 0, &rsrc);

> -		if (r) {

> -			pr_err("NUMA: bad reg property in memory node\n");

> -			break;

> +		for (i = 0; ; i++) {

> +			r = of_address_to_resource(np, i, &rsrc);

> +			if (r) {

> +				/* reached the end of of_address */

> +				if (i > 0) {

> +					r = 0;

> +					break;

> +				}

> +

> +				pr_err("NUMA: bad reg property in memory node\n");

> +				goto finished;

> +			}

> +

> +			r = numa_add_memblk(nid, rsrc.start,

> +					    rsrc.end - rsrc.start + 1);

> +			if (r)

> +				goto finished;

>  		}

> -

> -		r = numa_add_memblk(nid, rsrc.start,

> -				    rsrc.end - rsrc.start + 1);

> -		if (r)

> -			break;

>  	}

> +

> +finished:

>  	of_node_put(np);


This function can be simplified down to:

	for_each_node_by_type(np, "memory") {
		r = of_property_read_u32(np, "numa-node-id", &nid);
		if (r == -EINVAL)
			/*
			 * property doesn't exist if -EINVAL, continue
			 * looking for more memory nodes with
			 * "numa-node-id" property
			 */
			continue;
		else if (r)
			/* some other error */
			break;

		r = of_address_to_resource(np, 0, &rsrc);
		for (i = 0; !r; i++, r = of_address_to_resource(np, i, 
&rsrc)) {
			r = numa_add_memblk(nid, rsrc.start,
				    rsrc.end - rsrc.start + 1);
		}
	}
	of_node_put(np);

	return r;


Perhaps with a "if (!i && r) pr_err()" for an error message at the end.

Rob
--
To unsubscribe from this list: send the line "unsubscribe devicetree" in
the body of a message to majordomo@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html
Leizhen (ThunderTown) May 27, 2016, 3:36 a.m. UTC | #2
On 2016/5/26 21:13, Rob Herring wrote:
> On Thu, May 26, 2016 at 10:43:58AM +0800, Zhen Lei wrote:

>> For a normal memory@ devicetree node, its reg property can contains more

>> memory blocks.

>>

>> Because we don't known how many memory blocks maybe contained, so we try

>> from index=0, increase 1 until error returned(the end).

>>

>> Signed-off-by: Zhen Lei <thunder.leizhen@huawei.com>

>> ---

>>  drivers/of/of_numa.c | 30 ++++++++++++++++++++----------

>>  1 file changed, 20 insertions(+), 10 deletions(-)

>>

>> diff --git a/drivers/of/of_numa.c b/drivers/of/of_numa.c

>> index 21d831f..2c5f249 100644

>> --- a/drivers/of/of_numa.c

>> +++ b/drivers/of/of_numa.c

>> @@ -63,7 +63,7 @@ static int __init of_numa_parse_memory_nodes(void)

>>  	struct device_node *np = NULL;

>>  	struct resource rsrc;

>>  	u32 nid;

>> -	int r = 0;

>> +	int i, r = 0;

>>

>>  	for (;;) {

>>  		np = of_find_node_by_type(np, "memory");

>> @@ -82,17 +82,27 @@ static int __init of_numa_parse_memory_nodes(void)

>>  			/* some other error */

>>  			break;

>>

>> -		r = of_address_to_resource(np, 0, &rsrc);

>> -		if (r) {

>> -			pr_err("NUMA: bad reg property in memory node\n");

>> -			break;

>> +		for (i = 0; ; i++) {

>> +			r = of_address_to_resource(np, i, &rsrc);

>> +			if (r) {

>> +				/* reached the end of of_address */

>> +				if (i > 0) {

>> +					r = 0;

>> +					break;

>> +				}

>> +

>> +				pr_err("NUMA: bad reg property in memory node\n");

>> +				goto finished;

>> +			}

>> +

>> +			r = numa_add_memblk(nid, rsrc.start,

>> +					    rsrc.end - rsrc.start + 1);

>> +			if (r)

>> +				goto finished;

>>  		}

>> -

>> -		r = numa_add_memblk(nid, rsrc.start,

>> -				    rsrc.end - rsrc.start + 1);

>> -		if (r)

>> -			break;

>>  	}

>> +

>> +finished:

>>  	of_node_put(np);

> 

> This function can be simplified down to:

> 

> 	for_each_node_by_type(np, "memory") {

OK, That's good.

> 		r = of_property_read_u32(np, "numa-node-id", &nid);

> 		if (r == -EINVAL)

> 			/*

> 			 * property doesn't exist if -EINVAL, continue

> 			 * looking for more memory nodes with

> 			 * "numa-node-id" property

> 			 */

> 			continue;

Hi, everybody:
    If some "memory" node contains "numa-node-id", but some others missed. Can we simply ignored it?
I think we should break out too, and faking to only have node0.

> 		else if (r)

> 			/* some other error */

> 			break;

> 

> 		r = of_address_to_resource(np, 0, &rsrc);

> 		for (i = 0; !r; i++, r = of_address_to_resource(np, i, 


But r(non-zero) is just break this loop, the original is break the outer for (;;) loop

How about as below?

	for_each_node_by_type(np, "memory") {
		... ...

		for (i = 0; !of_address_to_resource(np, i, &rsrc); i++) {
                        r = numa_add_memblk(nid, rsrc.start,
                                            rsrc.end - rsrc.start + 1);
                        if (r)
                                goto finished;
                }

		if (!i)
			pr_err("NUMA: bad reg property in memory node\n");
	}

finished:
	

> &rsrc)) {

> 			r = numa_add_memblk(nid, rsrc.start,

> 				    rsrc.end - rsrc.start + 1);

> 		}

> 	}

> 	of_node_put(np);

> 

> 	return r;

> 

> 

> Perhaps with a "if (!i && r) pr_err()" for an error message at the end.

> 

> Rob

> 

> .

> 


--
To unsubscribe from this list: send the line "unsubscribe devicetree" in
the body of a message to majordomo@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html
Rob Herring (Arm) May 27, 2016, 4:20 a.m. UTC | #3
On Thu, May 26, 2016 at 10:36 PM, Leizhen (ThunderTown)
<thunder.leizhen@huawei.com> wrote:
>

>

> On 2016/5/26 21:13, Rob Herring wrote:

>> On Thu, May 26, 2016 at 10:43:58AM +0800, Zhen Lei wrote:

>>> For a normal memory@ devicetree node, its reg property can contains more

>>> memory blocks.

>>>

>>> Because we don't known how many memory blocks maybe contained, so we try

>>> from index=0, increase 1 until error returned(the end).

>>>

>>> Signed-off-by: Zhen Lei <thunder.leizhen@huawei.com>

>>> ---

>>>  drivers/of/of_numa.c | 30 ++++++++++++++++++++----------

>>>  1 file changed, 20 insertions(+), 10 deletions(-)

>>>

>>> diff --git a/drivers/of/of_numa.c b/drivers/of/of_numa.c

>>> index 21d831f..2c5f249 100644

>>> --- a/drivers/of/of_numa.c

>>> +++ b/drivers/of/of_numa.c

>>> @@ -63,7 +63,7 @@ static int __init of_numa_parse_memory_nodes(void)

>>>      struct device_node *np = NULL;

>>>      struct resource rsrc;

>>>      u32 nid;

>>> -    int r = 0;

>>> +    int i, r = 0;

>>>

>>>      for (;;) {

>>>              np = of_find_node_by_type(np, "memory");

>>> @@ -82,17 +82,27 @@ static int __init of_numa_parse_memory_nodes(void)

>>>                      /* some other error */

>>>                      break;

>>>

>>> -            r = of_address_to_resource(np, 0, &rsrc);

>>> -            if (r) {

>>> -                    pr_err("NUMA: bad reg property in memory node\n");

>>> -                    break;

>>> +            for (i = 0; ; i++) {

>>> +                    r = of_address_to_resource(np, i, &rsrc);

>>> +                    if (r) {

>>> +                            /* reached the end of of_address */

>>> +                            if (i > 0) {

>>> +                                    r = 0;

>>> +                                    break;

>>> +                            }

>>> +

>>> +                            pr_err("NUMA: bad reg property in memory node\n");

>>> +                            goto finished;

>>> +                    }

>>> +

>>> +                    r = numa_add_memblk(nid, rsrc.start,

>>> +                                        rsrc.end - rsrc.start + 1);

>>> +                    if (r)

>>> +                            goto finished;

>>>              }

>>> -

>>> -            r = numa_add_memblk(nid, rsrc.start,

>>> -                                rsrc.end - rsrc.start + 1);

>>> -            if (r)

>>> -                    break;

>>>      }

>>> +

>>> +finished:

>>>      of_node_put(np);

>>

>> This function can be simplified down to:

>>

>>       for_each_node_by_type(np, "memory") {

> OK, That's good.

>

>>               r = of_property_read_u32(np, "numa-node-id", &nid);

>>               if (r == -EINVAL)

>>                       /*

>>                        * property doesn't exist if -EINVAL, continue

>>                        * looking for more memory nodes with

>>                        * "numa-node-id" property

>>                        */

>>                       continue;

> Hi, everybody:

>     If some "memory" node contains "numa-node-id", but some others missed. Can we simply ignored it?

> I think we should break out too, and faking to only have node0.


Continuing to work is probably better than not.

>

>>               else if (r)

>>                       /* some other error */

>>                       break;

>>

>>               r = of_address_to_resource(np, 0, &rsrc);

>>               for (i = 0; !r; i++, r = of_address_to_resource(np, i,

>

> But r(non-zero) is just break this loop, the original is break the outer for (;;) loop


It is not really the kernel's job to validate the DT. If there's
random things in it then kernel's behavior is undefined.

>

> How about as below?

>

>         for_each_node_by_type(np, "memory") {

>                 ... ...

>

>                 for (i = 0; !of_address_to_resource(np, i, &rsrc); i++) {

>                         r = numa_add_memblk(nid, rsrc.start,

>                                             rsrc.end - rsrc.start + 1);

>                         if (r)

>                                 goto finished;

>                 }

>

>                 if (!i)

>                         pr_err("NUMA: bad reg property in memory node\n");

>         }

>

> finished:


Please try to avoid the goto. You can check r in the outer loop too.

Rob
--
To unsubscribe from this list: send the line "unsubscribe devicetree" in
the body of a message to majordomo@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html
Leizhen (ThunderTown) May 27, 2016, 7:04 a.m. UTC | #4
On 2016/5/27 12:20, Rob Herring wrote:
> On Thu, May 26, 2016 at 10:36 PM, Leizhen (ThunderTown)

> <thunder.leizhen@huawei.com> wrote:

>>

>>

>> On 2016/5/26 21:13, Rob Herring wrote:

>>> On Thu, May 26, 2016 at 10:43:58AM +0800, Zhen Lei wrote:

>>>> For a normal memory@ devicetree node, its reg property can contains more

>>>> memory blocks.

>>>>

>>>> Because we don't known how many memory blocks maybe contained, so we try

>>>> from index=0, increase 1 until error returned(the end).

>>>>

>>>> Signed-off-by: Zhen Lei <thunder.leizhen@huawei.com>

>>>> ---

>>>>  drivers/of/of_numa.c | 30 ++++++++++++++++++++----------

>>>>  1 file changed, 20 insertions(+), 10 deletions(-)

>>>>

>>>> diff --git a/drivers/of/of_numa.c b/drivers/of/of_numa.c

>>>> index 21d831f..2c5f249 100644

>>>> --- a/drivers/of/of_numa.c

>>>> +++ b/drivers/of/of_numa.c

>>>> @@ -63,7 +63,7 @@ static int __init of_numa_parse_memory_nodes(void)

>>>>      struct device_node *np = NULL;

>>>>      struct resource rsrc;

>>>>      u32 nid;

>>>> -    int r = 0;

>>>> +    int i, r = 0;

>>>>

>>>>      for (;;) {

>>>>              np = of_find_node_by_type(np, "memory");

>>>> @@ -82,17 +82,27 @@ static int __init of_numa_parse_memory_nodes(void)

>>>>                      /* some other error */

>>>>                      break;

>>>>

>>>> -            r = of_address_to_resource(np, 0, &rsrc);

>>>> -            if (r) {

>>>> -                    pr_err("NUMA: bad reg property in memory node\n");

>>>> -                    break;

>>>> +            for (i = 0; ; i++) {

>>>> +                    r = of_address_to_resource(np, i, &rsrc);

>>>> +                    if (r) {

>>>> +                            /* reached the end of of_address */

>>>> +                            if (i > 0) {

>>>> +                                    r = 0;

>>>> +                                    break;

>>>> +                            }

>>>> +

>>>> +                            pr_err("NUMA: bad reg property in memory node\n");

>>>> +                            goto finished;

>>>> +                    }

>>>> +

>>>> +                    r = numa_add_memblk(nid, rsrc.start,

>>>> +                                        rsrc.end - rsrc.start + 1);

>>>> +                    if (r)

>>>> +                            goto finished;

>>>>              }

>>>> -

>>>> -            r = numa_add_memblk(nid, rsrc.start,

>>>> -                                rsrc.end - rsrc.start + 1);

>>>> -            if (r)

>>>> -                    break;

>>>>      }

>>>> +

>>>> +finished:

>>>>      of_node_put(np);

>>>

>>> This function can be simplified down to:

>>>

>>>       for_each_node_by_type(np, "memory") {

>> OK, That's good.

>>

>>>               r = of_property_read_u32(np, "numa-node-id", &nid);

>>>               if (r == -EINVAL)

>>>                       /*

>>>                        * property doesn't exist if -EINVAL, continue

>>>                        * looking for more memory nodes with

>>>                        * "numa-node-id" property

>>>                        */

>>>                       continue;

>> Hi, everybody:

>>     If some "memory" node contains "numa-node-id", but some others missed. Can we simply ignored it?

>> I think we should break out too, and faking to only have node0.

> 

> Continuing to work is probably better than not.

> 

>>

>>>               else if (r)

>>>                       /* some other error */

>>>                       break;

>>>

>>>               r = of_address_to_resource(np, 0, &rsrc);

>>>               for (i = 0; !r; i++, r = of_address_to_resource(np, i,

>>

>> But r(non-zero) is just break this loop, the original is break the outer for (;;) loop

> 

> It is not really the kernel's job to validate the DT. If there's

> random things in it then kernel's behavior is undefined.

> 

>>

>> How about as below?

>>

>>         for_each_node_by_type(np, "memory") {

>>                 ... ...

>>

>>                 for (i = 0; !of_address_to_resource(np, i, &rsrc); i++) {

>>                         r = numa_add_memblk(nid, rsrc.start,

>>                                             rsrc.end - rsrc.start + 1);

>>                         if (r)

>>                                 goto finished;

>>                 }

>>

>>                 if (!i)

>>                         pr_err("NUMA: bad reg property in memory node\n");

>>         }

>>

>> finished:

> 

> Please try to avoid the goto. You can check r in the outer loop too.


OK. I have rewritten this function according to your advice.

	for_each_node_by_type(np, "memory") {
                r = of_property_read_u32(np, "numa-node-id", &nid);
                if (r == -EINVAL)
                        /*
                         * property doesn't exist if -EINVAL, continue
                         * looking for more memory nodes with
                         * "numa-node-id" property
                         */
                        continue;
									//I deleted the break of "some other error", and it will break in below "if (!i || r)" branch

                for (i = 0; !r && !of_address_to_resource(np, i, &rsrc); i++)
                        r = numa_add_memblk(nid, rsrc.start,
                                            rsrc.end - rsrc.start + 1);

                if (!i || r) {
                        of_node_put(np);				//I moved here, so that it looks more clear. Because in the normal use of for_each_node_by_type, of_node_put is not required
                        pr_err("NUMA: bad property in memory node\n");	//Deleted "reg", so that it's suitable or harmless for other error cases
                        break;
                }
        }

	return r;

> 

> Rob

> 

> .

> 


--
To unsubscribe from this list: send the line "unsubscribe devicetree" in
the body of a message to majordomo@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html
diff mbox

Patch

diff --git a/drivers/of/of_numa.c b/drivers/of/of_numa.c
index 21d831f..2c5f249 100644
--- a/drivers/of/of_numa.c
+++ b/drivers/of/of_numa.c
@@ -63,7 +63,7 @@  static int __init of_numa_parse_memory_nodes(void)
 	struct device_node *np = NULL;
 	struct resource rsrc;
 	u32 nid;
-	int r = 0;
+	int i, r = 0;

 	for (;;) {
 		np = of_find_node_by_type(np, "memory");
@@ -82,17 +82,27 @@  static int __init of_numa_parse_memory_nodes(void)
 			/* some other error */
 			break;

-		r = of_address_to_resource(np, 0, &rsrc);
-		if (r) {
-			pr_err("NUMA: bad reg property in memory node\n");
-			break;
+		for (i = 0; ; i++) {
+			r = of_address_to_resource(np, i, &rsrc);
+			if (r) {
+				/* reached the end of of_address */
+				if (i > 0) {
+					r = 0;
+					break;
+				}
+
+				pr_err("NUMA: bad reg property in memory node\n");
+				goto finished;
+			}
+
+			r = numa_add_memblk(nid, rsrc.start,
+					    rsrc.end - rsrc.start + 1);
+			if (r)
+				goto finished;
 		}
-
-		r = numa_add_memblk(nid, rsrc.start,
-				    rsrc.end - rsrc.start + 1);
-		if (r)
-			break;
 	}
+
+finished:
 	of_node_put(np);

 	return r;