Return-Path: Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S966309AbbKGBrW (ORCPT ); Fri, 6 Nov 2015 20:47:22 -0500 Received: from mail-bn1on0130.outbound.protection.outlook.com ([157.56.110.130]:46928 "EHLO na01-bn1-obe.outbound.protection.outlook.com" rhost-flags-OK-OK-OK-FAIL) by vger.kernel.org with ESMTP id S1758152AbbKGBq0 (ORCPT ); Fri, 6 Nov 2015 20:46:26 -0500 Authentication-Results: spf=fail (sender IP is 192.88.168.50) smtp.mailfrom=freescale.com; freescale.mail.onmicrosoft.com; dkim=none (message not signed) header.d=none;freescale.mail.onmicrosoft.com; dmarc=none action=none header.from=freescale.com; From: "J. German Rivera" To: , , , CC: , , , , , , , , , , , , , , "J. German Rivera" Subject: [PATCH v3 05/11] staging: fsl-mc: Extended MC bus allocator to include IRQs Date: Fri, 6 Nov 2015 19:30:21 -0600 Message-ID: <1446859827-6927-6-git-send-email-German.Rivera@freescale.com> X-Mailer: git-send-email 2.3.3 In-Reply-To: <1446859827-6927-1-git-send-email-German.Rivera@freescale.com> References: <1446859827-6927-1-git-send-email-German.Rivera@freescale.com> X-EOPAttributedMessage: 0 X-Microsoft-Exchange-Diagnostics: 1;BL2FFO11FD043;1:kYYzjjZDmwRT/xTvgxmVRA8Of/vJzOFeBwJ3YvidDOLS7lBRYoxpkYQttsvOLGVb6K7AtkHJiAVUz23B6FOqwAw5182nJfbb0r/7dUD2tGPR7dXh0pnDgZruVJoi1FIQu81v+y35jO8iwDjG0sbfnrCKzO7Q8Tprd+ZNkzp6wa5O5otot6wAU4UvYVzWe8YsVEykOKUQKSGO2IkxBdXMw8NcCKP5YkI1bQNt/b1dHvkZ3F4Kum9T+1G8iqzrfUpAzJ2vTMktDUIbArFR99YY7j/kO+P28b1v3H7oV3AJrvrxZKHKrVD7SoujX2TJyPFRUiOy7MZbajvxsBvaV3bZ6BVQvr7LNI22mYoQ6flKRxVH+sdBR44QY2Z/ufRoLr1p6xWkjJZl16U+/AIdYzH7EQ== X-Forefront-Antispam-Report: CIP:192.88.168.50;CTRY:US;IPV:NLI;EFV:NLI;SFV:NSPM;SFS:(10019020)(6009001)(2980300002)(1110001)(1109001)(339900001)(199003)(189002)(5890100001)(85426001)(4001430100002)(5003940100001)(106466001)(107886002)(97736004)(36756003)(81156007)(77096005)(5001770100001)(2950100001)(105606002)(229853001)(5001960100002)(50986999)(86362001)(76176999)(50226001)(2201001)(11100500001)(19580405001)(189998001)(47776003)(5008740100001)(6806005)(104016004)(50466002)(87936001)(19580395003)(92566002)(5007970100001)(48376002);DIR:OUT;SFP:1102;SCL:1;SRVR:BY2PR03MB505;H:tx30smr01.am.freescale.net;FPR:;SPF:Fail;PTR:InfoDomainNonexistent;A:1;MX:1;LANG:en; MIME-Version: 1.0 Content-Type: text/plain X-Microsoft-Exchange-Diagnostics: 1;BY2PR03MB505;2:7xQKBkTW3LNkkZWcn/HSi4EW9A8+R01FNbsT6cCLS2tXx4tQeiuF6MvRjMuPvmqD7G4gPFceXwoq1MfdHeGn5W12U7RIo06YzgJLmfqKaPx/+Aq6k/bmWNlerC6PtEz0pofOiOmn6J35ToPbqci/lnSqF7tucIu12d+TlQgUxJY=;3:w5twaa/alrRAeA0V9+bxO/CjenICXzeX2bdjDSweVH7VBk4dOZhO+nRvAdSLXzrlpKakryamBu3Q38j0SYAZnpQZVcwtn7XPjUoytcVN9KIorXXj79P+BeqP5kIvUu54prkvGOlnU2hp8fPMicHWDLvnZbH4YT7iowpEnEF94gn84RahcQ50ZaK4K64vtJIUYeXBLIC1gsujSnLnB2lBCtCb4LJtxPiZMfGLaKezRFA=;25:fmwowzfmQ1K8rlHiL6UDf/Z/AbANvgkiGz363SyJ0LCgmAnQC1M7Vi3LAXsZNHCauyRKP/cI1WksXWsWffh3iyTu5vdKdmthvA+w7lMioZAejtdD9I9xwHj7SBQnGNkSGWEr7pVY+Il2YSTg95sJijRTkGsLkfJQQNNaMBlG22z51wmbIWnOJVpenJ7cw5k/+bFPgGhYun5WovTETxSnBzr0b9D9kZEY5fZyPRKNCp1rHl+e98t8QBfqKl4wENWiD4vcDLyS0xakYkFW0EP9ng== X-Microsoft-Antispam: UriScan:;BCL:0;PCL:0;RULEID:;SRVR:BY2PR03MB505; X-Microsoft-Exchange-Diagnostics: 1;BY2PR03MB505;20:/pYkTt4W2CD2v91I8cwtF5dY74gS5BzJtwIWbpgB34ThCIRg2knhNxJ6SJH0tdA7ZE2N7iQaHaHWT1hLVJh1wJGppHyOw3rBNsv8d3GF/u9aDZ/0blksL8Pw0apkkmzRsmtH/K9fmorKb527dwX8cBHA4q4oVK+TE2HFc0lyfwMkcC5xUKJdzzByZhSQcYXf1kAVYqTG/sW6/yb9DFR7Cj7sDuVhircbu1gag7sYHLF+rdgLaeKeqoNGU5oGmpze8FuduaTdGh0LxNo5VBwcMQub6P6tPOXf4NqaoDu28qgmcdqLIEfZiPy81Kz7an+UOB7fvztmF7zoAWnlHfG2Nae+1H85YQuDsSfQU70Wp0I=;4:Re+ncOSJNIUE4JMhLjfzzlypU+vqmhqp4qKxtln0u5Wv+CcDLz3ixt6UHKN7T4CF6fLv2piYT46/iMQ+h6Zm1jm+NsmG+i277dfOHb1ul3tAYs6PMTZv0pGFM4BOuUOWCUA1MA93PGXfdKVd5sdu3Ru312teTY0Ofdub2tUFiJhGYZjw59b41ZysiUQHn5FbxoSkB3CPVdfwsPe4zxdPgaLa7NoBOAml4/eTCBdAGsiavoVB51sl7hrm1kpOa2tIFHrSglLY0kNW5Nrir+7cNgvjx9ychS9hfflfUdQj9vu8pPH0NF+W/snQKV6K0uLAMc2tHSrXCAU0Ho5fmcoH7LAfn5LU1xmiFVrGEGPads+h3zcWFNAsOcsRK98UQwch X-Microsoft-Antispam-PRVS: X-Exchange-Antispam-Report-Test: UriScan:(101931422205132); X-Exchange-Antispam-Report-CFA-Test: BCL:0;PCL:0;RULEID:(601004)(2401047)(520078)(5005006)(8121501046)(10201501046)(3002001);SRVR:BY2PR03MB505;BCL:0;PCL:0;RULEID:;SRVR:BY2PR03MB505; X-Forefront-PRVS: 0753EA505A X-Microsoft-Exchange-Diagnostics: =?us-ascii?Q?1;BY2PR03MB505;23:tOc+66a9oD2Ih9Q0bQmZNLiExbQ5FIhue1vhEOD0fN?= =?us-ascii?Q?j8/OhGmPtxT1fb9MPU2iAMEPyX8i8UzgJoTpujAeqfvGNl7iQUwJwBqf8Nm9?= =?us-ascii?Q?btXm+mITh37lLaaMDYBQNTTgTcdUeTdUymwoeYa55j/emiOlpEDzp1WwTzgp?= =?us-ascii?Q?BdEAREgus2pGR2khvu2ZPIMzDH/q0JG83+FE4cYAlH2fWcVNiwHrjzwFMTpg?= =?us-ascii?Q?W0T8lz10tkVFbkPxDeM+SZqHBWun8P9GG9EKTdJmNuTsZt6FRG1amx91K+pj?= =?us-ascii?Q?JwR6BVk1h1i0GUhd/ROKnAymVxygURbL3GIGdJK1w9XPZxz0/GEk0Td2AcN1?= =?us-ascii?Q?39xujkRM0GFBMlviWBs7zl9UPXzGNmNitdlZCxwnSNeYbT3iKPFS3pbvFF3W?= =?us-ascii?Q?cuYuEadDaoWvdESlls1fTTECF3DGC6UC/9S1RFDT8FzStMxOB9f8s7oIUdd0?= =?us-ascii?Q?kAjHpALMk8PwdqnEC5F+qKmLVz7Ih++qqc2hdTqWPItSKEugxY/Qy3yejECK?= =?us-ascii?Q?crGGyR9xEGZZrytR1hOc8jS6zXkOHOd9zeiyFIcgHFNHk+8biH9wons3Bm7+?= =?us-ascii?Q?TIfXD4hxfAGj5bjC03iOrdlfKFge7dXgu8kOrDjMkZNKGIJwLOmAPlKZZgh2?= =?us-ascii?Q?UcG/SjHJp4UIlx1eUS+ggHeiiuINVCdp0kxHeC5xUdYQ+cXYZhfllFXIRMy8?= =?us-ascii?Q?PSQf3KxJCpQKgZ7H7KWFR/0FHm8V49cjHzOlq1kBMQRuhN966JMv8sp90D9A?= =?us-ascii?Q?iA/tyHDu3EtwpSj5cLhtOjIwy6Vvus+BSMc6aDmLmLOnaDwGaZUj3nTWeo49?= =?us-ascii?Q?QwNnUWCpSSBsZe1G+m/LgU9Y9eBjFbDgXjzoQZgIhKEOgxOd/w6rUDGV89mo?= =?us-ascii?Q?e2Vwqm8Z3eHyRGMNmrJ5Qpbzqv04mx13CRalqSyNQJJ7yhYzKdJTfWzighd5?= =?us-ascii?Q?qNtLrRszlkpmPBr2rsm99YJWkppW6L72bcID4+JWvugP9lpNG23G0Ue6I8wd?= =?us-ascii?Q?uH3WspOONAHUuHGGfHOfXhbCV/sKR1T5XTgK0a8EPJ6jiDU4D9rSo5YguqWb?= =?us-ascii?Q?U2fDtsS/rV6T4wWs/xYhwZmnwRQiToJwzN0X4HhKPLAs8Njg=3D=3D?= X-Microsoft-Exchange-Diagnostics: 1;BY2PR03MB505;5:eGvThNTogVEZY9gi5DiJDPBTvKOyPc/s9LSJpUg/kqW3UpjhrWKzYm4TUiQm5LqlBX7qHRMpDW/fB0RkSIYSXik2Vf+8umIpxnFT5wxQqwNWtYKk+BSW78cXDNjbNfYh9UXJWaxxj5xumMxo1t/xuQ==;24:h2b9zFvYHNQbv2mU2BoSDb7Ve6gNDXxi1kouLJmZoy2sfrOiWDeuPHs8h55DTATPv1UZxnB+CxvyajX7hz3urJXrs1X1mhuDUw/SJIbiJek=;20:LedfH3Ha4daQtAr1uWzJaXiVTfuTkkqL+zTFajsQA0/1n1VMsEiJ6rqFTjhes34ovRGLdsMFxUDmz1tg3WZzlg== X-OriginatorOrg: freescale.com X-MS-Exchange-CrossTenant-OriginalArrivalTime: 07 Nov 2015 01:46:21.0684 (UTC) X-MS-Exchange-CrossTenant-Id: 710a03f5-10f6-4d38-9ff4-a80b81da590d X-MS-Exchange-CrossTenant-OriginalAttributedTenantConnectingIp: TenantId=710a03f5-10f6-4d38-9ff4-a80b81da590d;Ip=[192.88.168.50];Helo=[tx30smr01.am.freescale.net] X-MS-Exchange-CrossTenant-FromEntityHeader: HybridOnPrem X-MS-Exchange-Transport-CrossTenantHeadersStamped: BY2PR03MB505 Sender: linux-kernel-owner@vger.kernel.org List-ID: X-Mailing-List: linux-kernel@vger.kernel.org Content-Length: 10089 Lines: 339 All the IRQs for DPAA2 objects in the same DPRC must use the ICID of that DPRC, as their device Id in the GIC-ITS. Thus, all these IRQs must share the same ITT table in the GIC. As a result, a pool of IRQs with the same device Id must be preallocated per DPRC (fsl-mc bus instance). So, the fsl-mc bus object allocator is extended to also provide services to allocate IRQs to DPAA2 devices, from their parent fsl-mc bus IRQ pool. Signed-off-by: J. German Rivera --- CHANGE HISTORY Changes in v3: none Changes in v2: none drivers/staging/fsl-mc/bus/mc-allocator.c | 199 ++++++++++++++++++++++++++++ drivers/staging/fsl-mc/include/mc-private.h | 15 +++ drivers/staging/fsl-mc/include/mc.h | 9 ++ 3 files changed, 223 insertions(+) diff --git a/drivers/staging/fsl-mc/bus/mc-allocator.c b/drivers/staging/fsl-mc/bus/mc-allocator.c index 88d1857..c5fa628 100644 --- a/drivers/staging/fsl-mc/bus/mc-allocator.c +++ b/drivers/staging/fsl-mc/bus/mc-allocator.c @@ -15,6 +15,7 @@ #include "../include/dpcon-cmd.h" #include "dpmcp-cmd.h" #include "dpmcp.h" +#include /** * fsl_mc_resource_pool_add_device - add allocatable device to a resource @@ -160,6 +161,7 @@ static const char *const fsl_mc_pool_type_strings[] = { [FSL_MC_POOL_DPMCP] = "dpmcp", [FSL_MC_POOL_DPBP] = "dpbp", [FSL_MC_POOL_DPCON] = "dpcon", + [FSL_MC_POOL_IRQ] = "irq", }; static int __must_check object_type_to_pool_type(const char *object_type, @@ -465,6 +467,203 @@ void fsl_mc_object_free(struct fsl_mc_device *mc_adev) } EXPORT_SYMBOL_GPL(fsl_mc_object_free); +/* + * Initialize the interrupt pool associated with a MC bus. + * It allocates a block of IRQs from the GIC-ITS + */ +int fsl_mc_populate_irq_pool(struct fsl_mc_bus *mc_bus, + unsigned int irq_count) +{ + unsigned int i; + struct msi_desc *msi_desc; + struct fsl_mc_device_irq *irq_resources; + struct fsl_mc_device_irq *mc_dev_irq; + int error; + struct fsl_mc_device *mc_bus_dev = &mc_bus->mc_dev; + struct fsl_mc_resource_pool *res_pool = + &mc_bus->resource_pools[FSL_MC_POOL_IRQ]; + + if (WARN_ON(irq_count == 0 || + irq_count > FSL_MC_IRQ_POOL_MAX_TOTAL_IRQS)) + return -EINVAL; + + error = fsl_mc_msi_domain_alloc_irqs(&mc_bus_dev->dev, irq_count); + if (error < 0) + return error; + + irq_resources = devm_kzalloc(&mc_bus_dev->dev, + sizeof(*irq_resources) * irq_count, + GFP_KERNEL); + if (!irq_resources) { + error = -ENOMEM; + goto cleanup_msi_irqs; + } + + for (i = 0; i < irq_count; i++) { + mc_dev_irq = &irq_resources[i]; + + /* + * NOTE: This mc_dev_irq's MSI addr/value pair will be set + * by the fsl_mc_msi_write_msg() callback + */ + mc_dev_irq->resource.type = res_pool->type; + mc_dev_irq->resource.data = mc_dev_irq; + mc_dev_irq->resource.parent_pool = res_pool; + INIT_LIST_HEAD(&mc_dev_irq->resource.node); + list_add_tail(&mc_dev_irq->resource.node, &res_pool->free_list); + } + + for_each_msi_entry(msi_desc, &mc_bus_dev->dev) { + mc_dev_irq = &irq_resources[msi_desc->fsl_mc.msi_index]; + mc_dev_irq->msi_desc = msi_desc; + mc_dev_irq->resource.id = msi_desc->irq; + } + + res_pool->max_count = irq_count; + res_pool->free_count = irq_count; + mc_bus->irq_resources = irq_resources; + return 0; + +cleanup_msi_irqs: + fsl_mc_msi_domain_free_irqs(&mc_bus_dev->dev); + return error; +} +EXPORT_SYMBOL_GPL(fsl_mc_populate_irq_pool); + +/** + * Teardown the interrupt pool associated with an MC bus. + * It frees the IRQs that were allocated to the pool, back to the GIC-ITS. + */ +void fsl_mc_cleanup_irq_pool(struct fsl_mc_bus *mc_bus) +{ + struct fsl_mc_device *mc_bus_dev = &mc_bus->mc_dev; + struct fsl_mc_resource_pool *res_pool = + &mc_bus->resource_pools[FSL_MC_POOL_IRQ]; + + if (WARN_ON(!mc_bus->irq_resources)) + return; + + if (WARN_ON(res_pool->max_count == 0)) + return; + + if (WARN_ON(res_pool->free_count != res_pool->max_count)) + return; + + INIT_LIST_HEAD(&res_pool->free_list); + res_pool->max_count = 0; + res_pool->free_count = 0; + mc_bus->irq_resources = NULL; + fsl_mc_msi_domain_free_irqs(&mc_bus_dev->dev); +} +EXPORT_SYMBOL_GPL(fsl_mc_cleanup_irq_pool); + +/** + * It allocates the IRQs required by a given MC object device. The + * IRQs are allocated from the interrupt pool associated with the + * MC bus that contains the device, if the device is not a DPRC device. + * Otherwise, the IRQs are allocated from the interrupt pool associated + * with the MC bus that represents the DPRC device itself. + */ +int __must_check fsl_mc_allocate_irqs(struct fsl_mc_device *mc_dev) +{ + int i; + int irq_count; + int res_allocated_count = 0; + int error = -EINVAL; + struct fsl_mc_device_irq **irqs = NULL; + struct fsl_mc_bus *mc_bus; + struct fsl_mc_resource_pool *res_pool; + + if (WARN_ON(mc_dev->irqs)) + return -EINVAL; + + irq_count = mc_dev->obj_desc.irq_count; + if (WARN_ON(irq_count == 0)) + return -EINVAL; + + if (strcmp(mc_dev->obj_desc.type, "dprc") == 0) + mc_bus = to_fsl_mc_bus(mc_dev); + else + mc_bus = to_fsl_mc_bus(to_fsl_mc_device(mc_dev->dev.parent)); + + if (WARN_ON(!mc_bus->irq_resources)) + return -EINVAL; + + res_pool = &mc_bus->resource_pools[FSL_MC_POOL_IRQ]; + if (res_pool->free_count < irq_count) { + dev_err(&mc_dev->dev, + "Not able to allocate %u irqs for device\n", irq_count); + return -ENOSPC; + } + + irqs = devm_kzalloc(&mc_dev->dev, irq_count * sizeof(irqs[0]), + GFP_KERNEL); + if (!irqs) + return -ENOMEM; + + for (i = 0; i < irq_count; i++) { + struct fsl_mc_resource *resource; + + error = fsl_mc_resource_allocate(mc_bus, FSL_MC_POOL_IRQ, + &resource); + if (error < 0) + goto error_resource_alloc; + + irqs[i] = to_fsl_mc_irq(resource); + res_allocated_count++; + + WARN_ON(irqs[i]->mc_dev); + irqs[i]->mc_dev = mc_dev; + irqs[i]->dev_irq_index = i; + } + + mc_dev->irqs = irqs; + return 0; + +error_resource_alloc: + for (i = 0; i < res_allocated_count; i++) { + irqs[i]->mc_dev = NULL; + fsl_mc_resource_free(&irqs[i]->resource); + } + + return error; +} +EXPORT_SYMBOL_GPL(fsl_mc_allocate_irqs); + +/* + * It frees the IRQs that were allocated for a MC object device, by + * returning them to the corresponding interrupt pool. + */ +void fsl_mc_free_irqs(struct fsl_mc_device *mc_dev) +{ + int i; + int irq_count; + struct fsl_mc_bus *mc_bus; + struct fsl_mc_device_irq **irqs = mc_dev->irqs; + + if (WARN_ON(!irqs)) + return; + + irq_count = mc_dev->obj_desc.irq_count; + + if (strcmp(mc_dev->obj_desc.type, "dprc") == 0) + mc_bus = to_fsl_mc_bus(mc_dev); + else + mc_bus = to_fsl_mc_bus(to_fsl_mc_device(mc_dev->dev.parent)); + + if (WARN_ON(!mc_bus->irq_resources)) + return; + + for (i = 0; i < irq_count; i++) { + WARN_ON(!irqs[i]->mc_dev); + irqs[i]->mc_dev = NULL; + fsl_mc_resource_free(&irqs[i]->resource); + } + + mc_dev->irqs = NULL; +} +EXPORT_SYMBOL_GPL(fsl_mc_free_irqs); + /** * fsl_mc_allocator_probe - callback invoked when an allocatable device is * being added to the system diff --git a/drivers/staging/fsl-mc/include/mc-private.h b/drivers/staging/fsl-mc/include/mc-private.h index f2504bc..97295f0 100644 --- a/drivers/staging/fsl-mc/include/mc-private.h +++ b/drivers/staging/fsl-mc/include/mc-private.h @@ -30,6 +30,16 @@ struct irq_domain; struct msi_domain_info; /** + * Maximum number of total IRQs that can be pre-allocated for an MC bus' + * IRQ pool + */ +#define FSL_MC_IRQ_POOL_MAX_TOTAL_IRQS 256 + +struct device_node; +struct irq_domain; +struct msi_domain_info; + +/** * struct fsl_mc - Private data of a "fsl,qoriq-mc" platform device * @root_mc_bus_dev: MC object device representing the root DPRC * @num_translation_ranges: number of entries in addr_translation_ranges @@ -137,4 +147,9 @@ int __init its_fsl_mc_msi_init(void); void its_fsl_mc_msi_cleanup(void); +int fsl_mc_populate_irq_pool(struct fsl_mc_bus *mc_bus, + unsigned int irq_count); + +void fsl_mc_cleanup_irq_pool(struct fsl_mc_bus *mc_bus); + #endif /* _FSL_MC_PRIVATE_H_ */ diff --git a/drivers/staging/fsl-mc/include/mc.h b/drivers/staging/fsl-mc/include/mc.h index 1c1d6ae..ac7c1ce 100644 --- a/drivers/staging/fsl-mc/include/mc.h +++ b/drivers/staging/fsl-mc/include/mc.h @@ -14,12 +14,14 @@ #include #include #include +#include #include "../include/dprc.h" #define FSL_MC_VENDOR_FREESCALE 0x1957 struct fsl_mc_device; struct fsl_mc_io; +struct fsl_mc_bus; /** * struct fsl_mc_driver - MC object device driver object @@ -75,6 +77,7 @@ enum fsl_mc_pool_type { FSL_MC_POOL_DPMCP = 0x0, /* corresponds to "dpmcp" in the MC */ FSL_MC_POOL_DPBP, /* corresponds to "dpbp" in the MC */ FSL_MC_POOL_DPCON, /* corresponds to "dpcon" in the MC */ + FSL_MC_POOL_IRQ, /* * NOTE: New resource pool types must be added before this entry @@ -141,6 +144,7 @@ struct fsl_mc_device_irq { * NULL if none. * @obj_desc: MC description of the DPAA device * @regions: pointer to array of MMIO region entries + * @irqs: pointer to array of pointers to interrupts allocated to this device * @resource: generic resource associated with this MC object device, if any. * * Generic device object for MC object devices that are "attached" to a @@ -172,6 +176,7 @@ struct fsl_mc_device { struct fsl_mc_io *mc_io; struct dprc_obj_desc obj_desc; struct resource *regions; + struct fsl_mc_device_irq **irqs; struct fsl_mc_resource *resource; }; @@ -215,6 +220,10 @@ int __must_check fsl_mc_object_allocate(struct fsl_mc_device *mc_dev, void fsl_mc_object_free(struct fsl_mc_device *mc_adev); +int __must_check fsl_mc_allocate_irqs(struct fsl_mc_device *mc_dev); + +void fsl_mc_free_irqs(struct fsl_mc_device *mc_dev); + extern struct bus_type fsl_mc_bus_type; #endif /* _FSL_MC_H_ */ -- 2.3.3 -- To unsubscribe from this list: send the line "unsubscribe linux-kernel" in the body of a message to majordomo@vger.kernel.org More majordomo info at http://vger.kernel.org/majordomo-info.html Please read the FAQ at http://www.tux.org/lkml/