Return-Path: Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1752715AbaKGSHU (ORCPT ); Fri, 7 Nov 2014 13:07:20 -0500 Received: from mx1.redhat.com ([209.132.183.28]:51301 "EHLO mx1.redhat.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1751708AbaKGSHT (ORCPT ); Fri, 7 Nov 2014 13:07:19 -0500 Date: Fri, 7 Nov 2014 12:07:11 -0600 From: Seth Jennings To: Petr Mladek Cc: Josh Poimboeuf , Jiri Kosina , Vojtech Pavlik , Steven Rostedt , live-patching@vger.kernel.org, kpatch@redhat.com, linux-kernel@vger.kernel.org Subject: Re: module notifier: was Re: [PATCH 2/2] kernel: add support for live patching Message-ID: <20141107180711.GC2057@cerebellum.variantweb.net> References: <1415284748-14648-1-git-send-email-sjenning@redhat.com> <1415284748-14648-3-git-send-email-sjenning@redhat.com> <20141107171307.GC1136@dhcp128.suse.cz> MIME-Version: 1.0 Content-Type: text/plain; charset=us-ascii Content-Disposition: inline In-Reply-To: <20141107171307.GC1136@dhcp128.suse.cz> User-Agent: Mutt/1.5.23 (2014-03-12) Sender: linux-kernel-owner@vger.kernel.org List-ID: X-Mailing-List: linux-kernel@vger.kernel.org On Fri, Nov 07, 2014 at 06:13:07PM +0100, Petr Mladek wrote: > On Thu 2014-11-06 08:39:08, Seth Jennings wrote: > > This commit introduces code for the live patching core. It implements > > an ftrace-based mechanism and kernel interface for doing live patching > > of kernel and kernel module functions. > > > > It represents the greatest common functionality set between kpatch and > > kgraft and can accept patches built using either method. > > > > This first version does not implement any consistency mechanism that > > ensures that old and new code do not run together. In practice, ~90% of > > CVEs are safe to apply in this way, since they simply add a conditional > > check. However, any function change that can not execute safely with > > the old version of the function can _not_ be safely applied in this > > version. > > [...] > > > +/****************************** > > + * module notifier > > + *****************************/ > > + > > +static int lp_module_notify(struct notifier_block *nb, unsigned long action, > > + void *data) > > +{ > > + struct module *mod = data; > > + struct lpc_patch *patch; > > + struct lpc_object *obj; > > + int ret = 0; > > + > > + if (action != MODULE_STATE_COMING) > > + return 0; > > IMHO, we should handle also MODULE_STATE_GOING. We should unregister > the ftrace handlers and update the state of the affected objects > (ENABLED -> DISABLED) The mechanism we use to avoid this right now is taking a reference on patched module. We only release that reference after the patch is disabled, which unregisters all the patched functions from ftrace. However, your comment reminded me of an idea I had to use MODULE_STATE_GOING and let the lpc_mutex protect against races. I think it could be cleaner, but I haven't fleshed the idea out fully. > > > + down(&lpc_mutex); > > + > > + list_for_each_entry(patch, &lpc_patches, list) { > > + if (patch->state == DISABLED) > > + continue; > > + list_for_each_entry(obj, &patch->objs, list) { > > + if (strcmp(obj->name, mod->name)) > > + continue; > > + pr_notice("load of module '%s' detected, applying patch '%s'\n", > > + mod->name, patch->mod->name); > > + obj->mod = mod; > > + ret = lpc_enable_object(patch->mod, obj); > > + if (ret) > > + goto out; > > + break; > > + } > > + } > > + > > + up(&lpc_mutex); > > + return 0; > > +out: > > I would name this err_our or so to make it clear that it is used when > something fails. Just "err" good? > > > + up(&lpc_mutex); > > + WARN("failed to apply patch '%s' to module '%s'\n", > > + patch->mod->name, mod->name); > > + return 0; > > +} > > + > > +static struct notifier_block lp_module_nb = { > > + .notifier_call = lp_module_notify, > > + .priority = INT_MIN, /* called last */ > > The handler for MODULE_STATE_COMMING would need have higger priority, > if we want to cleanly unregister the ftrace handlers. Yes, we might need two handlers at different priorities if we decide to go that direction: one for MODULE_STATE_GOING at high/max and one for MODULE_STATE_COMING at low/min. Thanks, Seth > > Best Regards, > Petr -- 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/