Received: by 2002:ac0:a5a6:0:0:0:0:0 with SMTP id m35-v6csp59487imm; Fri, 21 Sep 2018 10:16:33 -0700 (PDT) X-Google-Smtp-Source: ANB0VdYJ5wHjqizWTu/wsnzsjuONRJDtacqFZZ8Fns4ag8s9ofKDvGYsaSU5jmROkHexNElQjAUJ X-Received: by 2002:a62:d709:: with SMTP id b9-v6mr15108727pfh.91.1537550193617; Fri, 21 Sep 2018 10:16:33 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1537550193; cv=none; d=google.com; s=arc-20160816; b=mRHOzYNMQhqAFAdjgufFf7sx1u4qs2ozbuQmOkAfen+VL1lW+2u0Btd7ZEWta7tlUJ HGSz7xxibyawDAQ0vzxc0s9aOK/j5gWBoZ8nJI2dPzfm42kkneYlYTV2UMVrD46Cp7gj KXA2u04kakxIBMAdgdTIFrBV/83wnuy06ipHFrzoCaULbpispm4FcwFK0/Yf/zK7Hq8K T1IOl3m/xKZLQFHMNG1BsvJsnXL4MHVsiVqtaTGSsvWfsu+7KGMbVxBZZmuTdMG5zKF4 gdkfVo68qJBHmBqUlM+RJtale4gYXHE3eJnQNXz88ORTEXX1pd2fnpxvmcHeJkCgUfE1 QWyw== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:sender:mime-version:content-transfer-encoding :spamdiagnosticmetadata:spamdiagnosticoutput:content-language :accept-language:message-id:date:thread-index:thread-topic:subject :cc:to:from:dkim-signature:dkim-signature; bh=MR/HQaUhm0XihkgqN6hYTuwkN+NabdaBFchiIvNKNtI=; b=f/g9td/lF7akj56/UlmwYH1UKyiSqB72Ge4xQP9w+eUN9NJRIV3SSGDyivmRTU87lf e7rJwtX/h2+QDCoUnvVQMYZcfIp2Ozg6avYGP/S6HGUvzhQLnW9TyP23p0rp5HwvKX7I K9NJLK7wtrvujvdyc4xG705/m90uykZ6MHDf0ltEAoLRrfnw5xvGkJixTmnaQHv8GxOF l6xa3Y/vTEfWUns4z/XkopUuN9BZMDMdEjXrkZxRJHh5qI3SrJMW9nI9pEoAlTT0iSAc 8itQw1Vlm3wM3kPwTtASROpSvtXb9sfhJyfYC9xRRqraq5EzxZpmzl6k9WfoHXmCTuTv fxFg== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@fb.com header.s=facebook header.b=YLkVrYDe; dkim=pass header.i=@fb.onmicrosoft.com header.s=selector1-fb-com header.b=PCqgrzl0; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=NONE sp=NONE dis=NONE) header.from=fb.com Return-Path: Received: from vger.kernel.org (vger.kernel.org. [209.132.180.67]) by mx.google.com with ESMTP id m29-v6si28945816pgl.304.2018.09.21.10.16.17; Fri, 21 Sep 2018 10:16:33 -0700 (PDT) Received-SPF: pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) client-ip=209.132.180.67; Authentication-Results: mx.google.com; dkim=pass header.i=@fb.com header.s=facebook header.b=YLkVrYDe; dkim=pass header.i=@fb.onmicrosoft.com header.s=selector1-fb-com header.b=PCqgrzl0; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=NONE sp=NONE dis=NONE) header.from=fb.com Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S2391115AbeIUXFF (ORCPT + 99 others); Fri, 21 Sep 2018 19:05:05 -0400 Received: from mx0b-00082601.pphosted.com ([67.231.153.30]:40852 "EHLO mx0a-00082601.pphosted.com" rhost-flags-OK-OK-OK-FAIL) by vger.kernel.org with ESMTP id S1728149AbeIUXE0 (ORCPT ); Fri, 21 Sep 2018 19:04:26 -0400 Received: from pps.filterd (m0001255.ppops.net [127.0.0.1]) by mx0b-00082601.pphosted.com (8.16.0.22/8.16.0.22) with SMTP id w8LH8jq3025307; Fri, 21 Sep 2018 10:14:14 -0700 DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=fb.com; h=from : to : cc : subject : date : message-id : content-type : content-transfer-encoding : mime-version; s=facebook; bh=MR/HQaUhm0XihkgqN6hYTuwkN+NabdaBFchiIvNKNtI=; b=YLkVrYDeL9Z/X6Lp4/DsCjpPc2hCXPXQKwxjJEvZBdzEXzE1u2YpDdV9EvwlIOOrlZ8K B5YK0ZIuWTQJU3U1irF9N3tWMNrmgA4uM/DLRu6m9+UR571zTS/ke22bPyIeplrICbmo Mu6HrwQcVLLT6xBLWvB/TVdOFS3qpaWX2o4= Received: from mail.thefacebook.com ([199.201.64.23]) by mx0b-00082601.pphosted.com with ESMTP id 2mn27brkv8-6 (version=TLSv1 cipher=ECDHE-RSA-AES256-SHA bits=256 verify=NOT); Fri, 21 Sep 2018 10:14:14 -0700 Received: from NAM05-DM3-obe.outbound.protection.outlook.com (192.168.54.28) by o365-in.thefacebook.com (192.168.16.24) with Microsoft SMTP Server (TLS) id 14.3.361.1; Fri, 21 Sep 2018 10:14:12 -0700 DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=fb.onmicrosoft.com; s=selector1-fb-com; h=From:Date:Subject:Message-ID:Content-Type:MIME-Version:X-MS-Exchange-SenderADCheck; bh=MR/HQaUhm0XihkgqN6hYTuwkN+NabdaBFchiIvNKNtI=; b=PCqgrzl0io3Kv9PcUrpAaObQ/nYd28jcSBgHavUC6Xp1qqD5OfuII0PZ0uRTSgLWyeXzzTIWVQXrjsznc8YTGemysFdyoeY/3SwSUxLZqXj3WjHb3piadcQhSUO8lsbMzCW1WUcEmcc60cwwvd2pZGRXCLYC+qnWls6HJCXQlaY= Received: from BY2PR15MB0167.namprd15.prod.outlook.com (10.163.64.141) by BY2PR15MB0232.namprd15.prod.outlook.com (10.163.64.157) with Microsoft SMTP Server (version=TLS1_2, cipher=TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384) id 15.20.1143.18; Fri, 21 Sep 2018 17:14:07 +0000 Received: from BY2PR15MB0167.namprd15.prod.outlook.com ([fe80::5c5b:75ea:cae:1e68]) by BY2PR15MB0167.namprd15.prod.outlook.com ([fe80::5c5b:75ea:cae:1e68%2]) with mapi id 15.20.1143.017; Fri, 21 Sep 2018 17:14:07 +0000 From: Roman Gushchin To: "netdev@vger.kernel.org" CC: "linux-kernel@vger.kernel.org" , Kernel Team , Roman Gushchin , Daniel Borkmann , Alexei Starovoitov Subject: [PATCH bpf-next 1/9] bpf: extend cgroup bpf core to allow multiple cgroup storage types Thread-Topic: [PATCH bpf-next 1/9] bpf: extend cgroup bpf core to allow multiple cgroup storage types Thread-Index: AQHUUc6AboXiYlzd7Uq1YROBHY3M3A== Date: Fri, 21 Sep 2018 17:14:07 +0000 Message-ID: <20180921171353.11050-1-guro@fb.com> Accept-Language: en-US Content-Language: en-US X-MS-Has-Attach: X-MS-TNEF-Correlator: x-clientproxiedby: MWHPR10CA0052.namprd10.prod.outlook.com (2603:10b6:300:2c::14) To BY2PR15MB0167.namprd15.prod.outlook.com (2a01:111:e400:58e0::13) x-ms-exchange-messagesentrepresentingtype: 1 x-originating-ip: [2620:10d:c090:200::7:15ea] x-ms-publictraffictype: Email x-microsoft-exchange-diagnostics: 1;BY2PR15MB0232;20:6e/wBbPFto1Q2XYYGTnvVTm1JPJWtZanwHuXvOJNWB4mcsJB6KP0m3vVi4Nm3aW8L9BEzrJ64hPkvLqNRL2XrJWJPxr0tO23Izj9jPQBikNH5k8SYhNFQcrvIwk6LyUUQB6ggN2dmg1rqiNKw4TnpEjxAgvpjCr+NRXNFzpfIRA= x-ms-office365-filtering-correlation-id: e060a562-b218-48f8-dfca-08d61fe5a311 x-microsoft-antispam: BCL:0;PCL:0;RULEID:(7020095)(4652040)(8989299)(4534165)(4627221)(201703031133081)(201702281549075)(8990200)(5600074)(711020)(2017052603328)(7153060)(7193020);SRVR:BY2PR15MB0232; x-ms-traffictypediagnostic: BY2PR15MB0232: x-microsoft-antispam-prvs: x-exchange-antispam-report-test: UriScan:(788757137089)(67672495146484); x-ms-exchange-senderadcheck: 1 x-exchange-antispam-report-cfa-test: BCL:0;PCL:0;RULEID:(8211001083)(6040522)(2401047)(8121501046)(5005006)(823302057)(3231355)(11241501184)(944501410)(52105095)(10201501046)(3002001)(93006095)(93001095)(149027)(150027)(6041310)(201703131423095)(201702281528075)(20161123555045)(201703061421075)(201703061406153)(20161123564045)(20161123560045)(20161123558120)(20161123562045)(201708071742011)(7699051);SRVR:BY2PR15MB0232;BCL:0;PCL:0;RULEID:;SRVR:BY2PR15MB0232; x-forefront-prvs: 0802ADD973 x-forefront-antispam-report: SFV:NSPM;SFS:(10019020)(346002)(39860400002)(376002)(136003)(396003)(366004)(199004)(189003)(386003)(5640700003)(14444005)(186003)(86362001)(486006)(6512007)(36756003)(99286004)(256004)(5024004)(6436002)(6486002)(5250100002)(54906003)(6916009)(71200400001)(71190400001)(2900100001)(476003)(575784001)(102836004)(2616005)(52116002)(8676002)(46003)(81166006)(305945005)(6506007)(106356001)(25786009)(2351001)(5660300001)(6116002)(53946003)(97736004)(81156014)(53936002)(316002)(2906002)(4326008)(1730700003)(1076002)(478600001)(7736002)(68736007)(2501003)(105586002)(8936002)(14454004)(42262002);DIR:OUT;SFP:1102;SCL:1;SRVR:BY2PR15MB0232;H:BY2PR15MB0167.namprd15.prod.outlook.com;FPR:;SPF:None;LANG:en;PTR:InfoNoRecords;A:1;MX:1; received-spf: None (protection.outlook.com: fb.com does not designate permitted sender hosts) x-microsoft-antispam-message-info: YTf7ZTkiQ03EsIh5U2BfMsfftVo2408M+rO1tCVHtOnjxea3xx29rxR79zzc6qWyfDxIqZsvku0CVmKE62B91RUZduJPlX/MVXGB8OQ/Tt0rUMF+/kKjDYOpCFj9dN58dYJmdrzzMCohyC0TBoJRNEC7cgW4O9D1k9Esv1cizp56zRgJowsr29MSqozvTnS6Ga6sN6w50LXm1J8/EcPevXQVNqvk6IhRsJ+zD25ptCRVvz+NAmqSKRH4htUFNSZZKz5zjG/CroSDz5jQHAx56hFSrPI0EMbM39bPGk4vrpQNLdPMhbRAhNdrTjQLpq62bcPRdHV8iEERhxagSKDmG2fMZ4BUPuYp/YKZ32LFkKw= spamdiagnosticoutput: 1:99 spamdiagnosticmetadata: NSPM Content-Type: text/plain; charset="iso-8859-1" Content-Transfer-Encoding: quoted-printable MIME-Version: 1.0 X-MS-Exchange-CrossTenant-Network-Message-Id: e060a562-b218-48f8-dfca-08d61fe5a311 X-MS-Exchange-CrossTenant-originalarrivaltime: 21 Sep 2018 17:14:07.2240 (UTC) X-MS-Exchange-CrossTenant-fromentityheader: Hosted X-MS-Exchange-CrossTenant-id: 8ae927fe-1255-47a7-a2af-5f3a069daaa2 X-MS-Exchange-Transport-CrossTenantHeadersStamped: BY2PR15MB0232 X-OriginatorOrg: fb.com X-Proofpoint-Virus-Version: vendor=fsecure engine=2.50.10434:,, definitions=2018-09-21_07:,, signatures=0 X-Proofpoint-Spam-Reason: safe X-FB-Internal: Safe Sender: linux-kernel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org In order to introduce per-cpu cgroup storage, let's generalize bpf cgroup core to support multiple cgroup storage types. Potentially, per-node cgroup storage can be added later. This commit is mostly a formal change that replaces cgroup_storage pointer with a array of cgroup_storage pointers. It doesn't actually introduce a new storage type, it will be done later. Each bpf program is now able to have one cgroup storage of each type. Signed-off-by: Roman Gushchin Cc: Daniel Borkmann Cc: Alexei Starovoitov --- include/linux/bpf-cgroup.h | 38 ++++++++++++++------ include/linux/bpf.h | 11 ++++-- kernel/bpf/cgroup.c | 74 ++++++++++++++++++++++++++------------ kernel/bpf/helpers.c | 15 ++++---- kernel/bpf/local_storage.c | 18 ++++++---- kernel/bpf/syscall.c | 9 +++-- kernel/bpf/verifier.c | 8 +++-- net/bpf/test_run.c | 20 +++++++---- 8 files changed, 136 insertions(+), 57 deletions(-) diff --git a/include/linux/bpf-cgroup.h b/include/linux/bpf-cgroup.h index f91b0f8ff3a9..e9871b012dac 100644 --- a/include/linux/bpf-cgroup.h +++ b/include/linux/bpf-cgroup.h @@ -2,6 +2,7 @@ #ifndef _BPF_CGROUP_H #define _BPF_CGROUP_H =20 +#include #include #include #include @@ -22,7 +23,10 @@ struct bpf_cgroup_storage; extern struct static_key_false cgroup_bpf_enabled_key; #define cgroup_bpf_enabled static_branch_unlikely(&cgroup_bpf_enabled_key) =20 -DECLARE_PER_CPU(void*, bpf_cgroup_storage); +DECLARE_PER_CPU(void*, bpf_cgroup_storage[MAX_BPF_CGROUP_STORAGE_TYPE]); + +#define for_each_cgroup_storage_type(stype) \ + for (stype =3D 0; stype < MAX_BPF_CGROUP_STORAGE_TYPE; stype++) =20 struct bpf_cgroup_storage_map; =20 @@ -43,7 +47,7 @@ struct bpf_cgroup_storage { struct bpf_prog_list { struct list_head node; struct bpf_prog *prog; - struct bpf_cgroup_storage *storage; + struct bpf_cgroup_storage *storage[MAX_BPF_CGROUP_STORAGE_TYPE]; }; =20 struct bpf_prog_array; @@ -101,18 +105,29 @@ int __cgroup_bpf_run_filter_sock_ops(struct sock *sk, int __cgroup_bpf_check_dev_permission(short dev_type, u32 major, u32 minor= , short access, enum bpf_attach_type type); =20 -static inline void bpf_cgroup_storage_set(struct bpf_cgroup_storage *stora= ge) +static inline enum bpf_cgroup_storage_type cgroup_storage_type( + struct bpf_map *map) { + return BPF_CGROUP_STORAGE_SHARED; +} + +static inline void bpf_cgroup_storage_set(struct bpf_cgroup_storage + *storage[MAX_BPF_CGROUP_STORAGE_TYPE]) +{ + enum bpf_cgroup_storage_type stype; struct bpf_storage_buffer *buf; =20 - if (!storage) - return; + for_each_cgroup_storage_type(stype) { + if (!storage[stype]) + continue; =20 - buf =3D READ_ONCE(storage->buf); - this_cpu_write(bpf_cgroup_storage, &buf->data[0]); + buf =3D READ_ONCE(storage[stype]->buf); + this_cpu_write(bpf_cgroup_storage[stype], &buf->data[0]); + } } =20 -struct bpf_cgroup_storage *bpf_cgroup_storage_alloc(struct bpf_prog *prog)= ; +struct bpf_cgroup_storage *bpf_cgroup_storage_alloc(struct bpf_prog *prog, + enum bpf_cgroup_storage_type stype); void bpf_cgroup_storage_free(struct bpf_cgroup_storage *storage); void bpf_cgroup_storage_link(struct bpf_cgroup_storage *storage, struct cgroup *cgroup, @@ -265,13 +280,14 @@ static inline int cgroup_bpf_prog_query(const union b= pf_attr *attr, return -EINVAL; } =20 -static inline void bpf_cgroup_storage_set(struct bpf_cgroup_storage *stora= ge) {} +static inline void bpf_cgroup_storage_set( + struct bpf_cgroup_storage *storage[MAX_BPF_CGROUP_STORAGE_TYPE]) {} static inline int bpf_cgroup_storage_assign(struct bpf_prog *prog, struct bpf_map *map) { return 0; } static inline void bpf_cgroup_storage_release(struct bpf_prog *prog, struct bpf_map *map) {} static inline struct bpf_cgroup_storage *bpf_cgroup_storage_alloc( - struct bpf_prog *prog) { return 0; } + struct bpf_prog *prog, enum bpf_cgroup_storage_type stype) { return 0; } static inline void bpf_cgroup_storage_free( struct bpf_cgroup_storage *storage) {} =20 @@ -293,6 +309,8 @@ static inline void bpf_cgroup_storage_free( #define BPF_CGROUP_RUN_PROG_SOCK_OPS(sock_ops) ({ 0; }) #define BPF_CGROUP_RUN_PROG_DEVICE_CGROUP(type,major,minor,access) ({ 0; }= ) =20 +#define for_each_cgroup_storage_type(stype) for (; false; ) + #endif /* CONFIG_CGROUP_BPF */ =20 #endif /* _BPF_CGROUP_H */ diff --git a/include/linux/bpf.h b/include/linux/bpf.h index 988a00797bcd..b457fbe7b70b 100644 --- a/include/linux/bpf.h +++ b/include/linux/bpf.h @@ -272,6 +272,13 @@ struct bpf_prog_offload { u32 jited_len; }; =20 +enum bpf_cgroup_storage_type { + BPF_CGROUP_STORAGE_SHARED, + __BPF_CGROUP_STORAGE_MAX +}; + +#define MAX_BPF_CGROUP_STORAGE_TYPE __BPF_CGROUP_STORAGE_MAX + struct bpf_prog_aux { atomic_t refcnt; u32 used_map_cnt; @@ -289,7 +296,7 @@ struct bpf_prog_aux { struct bpf_prog *prog; struct user_struct *user; u64 load_time; /* ns since boottime */ - struct bpf_map *cgroup_storage; + struct bpf_map *cgroup_storage[MAX_BPF_CGROUP_STORAGE_TYPE]; char name[BPF_OBJ_NAME_LEN]; #ifdef CONFIG_SECURITY void *security; @@ -358,7 +365,7 @@ int bpf_prog_test_run_skb(struct bpf_prog *prog, const = union bpf_attr *kattr, */ struct bpf_prog_array_item { struct bpf_prog *prog; - struct bpf_cgroup_storage *cgroup_storage; + struct bpf_cgroup_storage *cgroup_storage[MAX_BPF_CGROUP_STORAGE_TYPE]; }; =20 struct bpf_prog_array { diff --git a/kernel/bpf/cgroup.c b/kernel/bpf/cgroup.c index 6a7d931bbc55..065c3d9ff8eb 100644 --- a/kernel/bpf/cgroup.c +++ b/kernel/bpf/cgroup.c @@ -25,6 +25,7 @@ EXPORT_SYMBOL(cgroup_bpf_enabled_key); */ void cgroup_bpf_put(struct cgroup *cgrp) { + enum bpf_cgroup_storage_type stype; unsigned int type; =20 for (type =3D 0; type < ARRAY_SIZE(cgrp->bpf.progs); type++) { @@ -34,8 +35,10 @@ void cgroup_bpf_put(struct cgroup *cgrp) list_for_each_entry_safe(pl, tmp, progs, node) { list_del(&pl->node); bpf_prog_put(pl->prog); - bpf_cgroup_storage_unlink(pl->storage); - bpf_cgroup_storage_free(pl->storage); + for_each_cgroup_storage_type(stype) { + bpf_cgroup_storage_unlink(pl->storage[stype]); + bpf_cgroup_storage_free(pl->storage[stype]); + } kfree(pl); static_branch_dec(&cgroup_bpf_enabled_key); } @@ -97,6 +100,7 @@ static int compute_effective_progs(struct cgroup *cgrp, enum bpf_attach_type type, struct bpf_prog_array __rcu **array) { + enum bpf_cgroup_storage_type stype; struct bpf_prog_array *progs; struct bpf_prog_list *pl; struct cgroup *p =3D cgrp; @@ -125,7 +129,9 @@ static int compute_effective_progs(struct cgroup *cgrp, continue; =20 progs->items[cnt].prog =3D pl->prog; - progs->items[cnt].cgroup_storage =3D pl->storage; + for_each_cgroup_storage_type(stype) + progs->items[cnt].cgroup_storage[stype] =3D + pl->storage[stype]; cnt++; } } while ((p =3D cgroup_parent(p))); @@ -232,7 +238,9 @@ int __cgroup_bpf_attach(struct cgroup *cgrp, struct bpf= _prog *prog, { struct list_head *progs =3D &cgrp->bpf.progs[type]; struct bpf_prog *old_prog =3D NULL; - struct bpf_cgroup_storage *storage, *old_storage =3D NULL; + struct bpf_cgroup_storage *storage[MAX_BPF_CGROUP_STORAGE_TYPE], + *old_storage[MAX_BPF_CGROUP_STORAGE_TYPE] =3D {NULL}; + enum bpf_cgroup_storage_type stype; struct bpf_prog_list *pl; bool pl_was_allocated; int err; @@ -254,34 +262,44 @@ int __cgroup_bpf_attach(struct cgroup *cgrp, struct b= pf_prog *prog, if (prog_list_length(progs) >=3D BPF_CGROUP_MAX_PROGS) return -E2BIG; =20 - storage =3D bpf_cgroup_storage_alloc(prog); - if (IS_ERR(storage)) - return -ENOMEM; + for_each_cgroup_storage_type(stype) { + storage[stype] =3D bpf_cgroup_storage_alloc(prog, stype); + if (IS_ERR(storage[stype])) { + storage[stype] =3D NULL; + for_each_cgroup_storage_type(stype) + bpf_cgroup_storage_free(storage[stype]); + return -ENOMEM; + } + } =20 if (flags & BPF_F_ALLOW_MULTI) { list_for_each_entry(pl, progs, node) { if (pl->prog =3D=3D prog) { /* disallow attaching the same prog twice */ - bpf_cgroup_storage_free(storage); + for_each_cgroup_storage_type(stype) + bpf_cgroup_storage_free(storage[stype]); return -EINVAL; } } =20 pl =3D kmalloc(sizeof(*pl), GFP_KERNEL); if (!pl) { - bpf_cgroup_storage_free(storage); + for_each_cgroup_storage_type(stype) + bpf_cgroup_storage_free(storage[stype]); return -ENOMEM; } =20 pl_was_allocated =3D true; pl->prog =3D prog; - pl->storage =3D storage; + for_each_cgroup_storage_type(stype) + pl->storage[stype] =3D storage[stype]; list_add_tail(&pl->node, progs); } else { if (list_empty(progs)) { pl =3D kmalloc(sizeof(*pl), GFP_KERNEL); if (!pl) { - bpf_cgroup_storage_free(storage); + for_each_cgroup_storage_type(stype) + bpf_cgroup_storage_free(storage[stype]); return -ENOMEM; } pl_was_allocated =3D true; @@ -289,12 +307,15 @@ int __cgroup_bpf_attach(struct cgroup *cgrp, struct b= pf_prog *prog, } else { pl =3D list_first_entry(progs, typeof(*pl), node); old_prog =3D pl->prog; - old_storage =3D pl->storage; - bpf_cgroup_storage_unlink(old_storage); + for_each_cgroup_storage_type(stype) { + old_storage[stype] =3D pl->storage[stype]; + bpf_cgroup_storage_unlink(old_storage[stype]); + } pl_was_allocated =3D false; } pl->prog =3D prog; - pl->storage =3D storage; + for_each_cgroup_storage_type(stype) + pl->storage[stype] =3D storage[stype]; } =20 cgrp->bpf.flags[type] =3D flags; @@ -304,21 +325,27 @@ int __cgroup_bpf_attach(struct cgroup *cgrp, struct b= pf_prog *prog, goto cleanup; =20 static_branch_inc(&cgroup_bpf_enabled_key); - if (old_storage) - bpf_cgroup_storage_free(old_storage); + for_each_cgroup_storage_type(stype) { + if (!old_storage[stype]) + continue; + bpf_cgroup_storage_free(old_storage[stype]); + } if (old_prog) { bpf_prog_put(old_prog); static_branch_dec(&cgroup_bpf_enabled_key); } - bpf_cgroup_storage_link(storage, cgrp, type); + for_each_cgroup_storage_type(stype) + bpf_cgroup_storage_link(storage[stype], cgrp, type); return 0; =20 cleanup: /* and cleanup the prog list */ pl->prog =3D old_prog; - bpf_cgroup_storage_free(pl->storage); - pl->storage =3D old_storage; - bpf_cgroup_storage_link(old_storage, cgrp, type); + for_each_cgroup_storage_type(stype) { + bpf_cgroup_storage_free(pl->storage[stype]); + pl->storage[stype] =3D old_storage[stype]; + bpf_cgroup_storage_link(old_storage[stype], cgrp, type); + } if (pl_was_allocated) { list_del(&pl->node); kfree(pl); @@ -339,6 +366,7 @@ int __cgroup_bpf_detach(struct cgroup *cgrp, struct bpf= _prog *prog, enum bpf_attach_type type, u32 unused_flags) { struct list_head *progs =3D &cgrp->bpf.progs[type]; + enum bpf_cgroup_storage_type stype; u32 flags =3D cgrp->bpf.flags[type]; struct bpf_prog *old_prog =3D NULL; struct bpf_prog_list *pl; @@ -385,8 +413,10 @@ int __cgroup_bpf_detach(struct cgroup *cgrp, struct bp= f_prog *prog, =20 /* now can actually delete it from this cgroup list */ list_del(&pl->node); - bpf_cgroup_storage_unlink(pl->storage); - bpf_cgroup_storage_free(pl->storage); + for_each_cgroup_storage_type(stype) { + bpf_cgroup_storage_unlink(pl->storage[stype]); + bpf_cgroup_storage_free(pl->storage[stype]); + } kfree(pl); if (list_empty(progs)) /* last program was detached, reset flags to zero */ diff --git a/kernel/bpf/helpers.c b/kernel/bpf/helpers.c index 1991466b8327..9070b2ace6aa 100644 --- a/kernel/bpf/helpers.c +++ b/kernel/bpf/helpers.c @@ -194,16 +194,18 @@ const struct bpf_func_proto bpf_get_current_cgroup_id= _proto =3D { .ret_type =3D RET_INTEGER, }; =20 -DECLARE_PER_CPU(void*, bpf_cgroup_storage); +#ifdef CONFIG_CGROUP_BPF +DECLARE_PER_CPU(void*, bpf_cgroup_storage[MAX_BPF_CGROUP_STORAGE_TYPE]); =20 BPF_CALL_2(bpf_get_local_storage, struct bpf_map *, map, u64, flags) { - /* map and flags arguments are not used now, - * but provide an ability to extend the API - * for other types of local storages. - * verifier checks that their values are correct. + /* flags argument is not used now, + * but provides an ability to extend the API. + * verifier checks that its value is correct. */ - return (unsigned long) this_cpu_read(bpf_cgroup_storage); + enum bpf_cgroup_storage_type stype =3D cgroup_storage_type(map); + + return (unsigned long) this_cpu_read(bpf_cgroup_storage[stype]); } =20 const struct bpf_func_proto bpf_get_local_storage_proto =3D { @@ -214,3 +216,4 @@ const struct bpf_func_proto bpf_get_local_storage_proto= =3D { .arg2_type =3D ARG_ANYTHING, }; #endif +#endif diff --git a/kernel/bpf/local_storage.c b/kernel/bpf/local_storage.c index 22ad967d1e5f..0bd9f19fc557 100644 --- a/kernel/bpf/local_storage.c +++ b/kernel/bpf/local_storage.c @@ -7,7 +7,7 @@ #include #include =20 -DEFINE_PER_CPU(void*, bpf_cgroup_storage); +DEFINE_PER_CPU(void*, bpf_cgroup_storage[MAX_BPF_CGROUP_STORAGE_TYPE]); =20 #ifdef CONFIG_CGROUP_BPF =20 @@ -251,6 +251,7 @@ const struct bpf_map_ops cgroup_storage_map_ops =3D { =20 int bpf_cgroup_storage_assign(struct bpf_prog *prog, struct bpf_map *_map) { + enum bpf_cgroup_storage_type stype =3D cgroup_storage_type(_map); struct bpf_cgroup_storage_map *map =3D map_to_storage(_map); int ret =3D -EBUSY; =20 @@ -258,11 +259,12 @@ int bpf_cgroup_storage_assign(struct bpf_prog *prog, = struct bpf_map *_map) =20 if (map->prog && map->prog !=3D prog) goto unlock; - if (prog->aux->cgroup_storage && prog->aux->cgroup_storage !=3D _map) + if (prog->aux->cgroup_storage[stype] && + prog->aux->cgroup_storage[stype] !=3D _map) goto unlock; =20 map->prog =3D prog; - prog->aux->cgroup_storage =3D _map; + prog->aux->cgroup_storage[stype] =3D _map; ret =3D 0; unlock: spin_unlock_bh(&map->lock); @@ -272,24 +274,26 @@ int bpf_cgroup_storage_assign(struct bpf_prog *prog, = struct bpf_map *_map) =20 void bpf_cgroup_storage_release(struct bpf_prog *prog, struct bpf_map *_ma= p) { + enum bpf_cgroup_storage_type stype =3D cgroup_storage_type(_map); struct bpf_cgroup_storage_map *map =3D map_to_storage(_map); =20 spin_lock_bh(&map->lock); if (map->prog =3D=3D prog) { - WARN_ON(prog->aux->cgroup_storage !=3D _map); + WARN_ON(prog->aux->cgroup_storage[stype] !=3D _map); map->prog =3D NULL; - prog->aux->cgroup_storage =3D NULL; + prog->aux->cgroup_storage[stype] =3D NULL; } spin_unlock_bh(&map->lock); } =20 -struct bpf_cgroup_storage *bpf_cgroup_storage_alloc(struct bpf_prog *prog) +struct bpf_cgroup_storage *bpf_cgroup_storage_alloc(struct bpf_prog *prog, + enum bpf_cgroup_storage_type stype) { struct bpf_cgroup_storage *storage; struct bpf_map *map; u32 pages; =20 - map =3D prog->aux->cgroup_storage; + map =3D prog->aux->cgroup_storage[stype]; if (!map) return NULL; =20 diff --git a/kernel/bpf/syscall.c b/kernel/bpf/syscall.c index b3c2d09bcf7a..8c91d2b41b1e 100644 --- a/kernel/bpf/syscall.c +++ b/kernel/bpf/syscall.c @@ -988,10 +988,15 @@ static int find_prog_type(enum bpf_prog_type type, st= ruct bpf_prog *prog) /* drop refcnt on maps used by eBPF program and free auxilary data */ static void free_used_maps(struct bpf_prog_aux *aux) { + enum bpf_cgroup_storage_type stype; int i; =20 - if (aux->cgroup_storage) - bpf_cgroup_storage_release(aux->prog, aux->cgroup_storage); + for_each_cgroup_storage_type(stype) { + if (!aux->cgroup_storage[stype]) + continue; + bpf_cgroup_storage_release(aux->prog, + aux->cgroup_storage[stype]); + } =20 for (i =3D 0; i < aux->used_map_cnt; i++) bpf_map_put(aux->used_maps[i]); diff --git a/kernel/bpf/verifier.c b/kernel/bpf/verifier.c index 8ccbff4fff93..e75f36de91d6 100644 --- a/kernel/bpf/verifier.c +++ b/kernel/bpf/verifier.c @@ -5171,11 +5171,15 @@ static int replace_map_fd_with_map_ptr(struct bpf_v= erifier_env *env) /* drop refcnt of maps used by the rejected program */ static void release_maps(struct bpf_verifier_env *env) { + enum bpf_cgroup_storage_type stype; int i; =20 - if (env->prog->aux->cgroup_storage) + for_each_cgroup_storage_type(stype) { + if (!env->prog->aux->cgroup_storage[stype]) + continue; bpf_cgroup_storage_release(env->prog, - env->prog->aux->cgroup_storage); + env->prog->aux->cgroup_storage[stype]); + } =20 for (i =3D 0; i < env->used_map_cnt; i++) bpf_map_put(env->used_maps[i]); diff --git a/net/bpf/test_run.c b/net/bpf/test_run.c index f4078830ea50..0c423b8cd75c 100644 --- a/net/bpf/test_run.c +++ b/net/bpf/test_run.c @@ -12,7 +12,7 @@ #include =20 static __always_inline u32 bpf_test_run_one(struct bpf_prog *prog, void *c= tx, - struct bpf_cgroup_storage *storage) + struct bpf_cgroup_storage *storage[MAX_BPF_CGROUP_STORAGE_TYPE]) { u32 ret; =20 @@ -28,13 +28,20 @@ static __always_inline u32 bpf_test_run_one(struct bpf_= prog *prog, void *ctx, =20 static u32 bpf_test_run(struct bpf_prog *prog, void *ctx, u32 repeat, u32 = *time) { - struct bpf_cgroup_storage *storage =3D NULL; + struct bpf_cgroup_storage *storage[MAX_BPF_CGROUP_STORAGE_TYPE] =3D { 0 }= ; + enum bpf_cgroup_storage_type stype; u64 time_start, time_spent =3D 0; u32 ret =3D 0, i; =20 - storage =3D bpf_cgroup_storage_alloc(prog); - if (IS_ERR(storage)) - return PTR_ERR(storage); + for_each_cgroup_storage_type(stype) { + storage[stype] =3D bpf_cgroup_storage_alloc(prog, stype); + if (IS_ERR(storage[stype])) { + storage[stype] =3D NULL; + for_each_cgroup_storage_type(stype) + bpf_cgroup_storage_free(storage[stype]); + return -ENOMEM; + } + } =20 if (!repeat) repeat =3D 1; @@ -53,7 +60,8 @@ static u32 bpf_test_run(struct bpf_prog *prog, void *ctx,= u32 repeat, u32 *time) do_div(time_spent, repeat); *time =3D time_spent > U32_MAX ? U32_MAX : (u32)time_spent; =20 - bpf_cgroup_storage_free(storage); + for_each_cgroup_storage_type(stype) + bpf_cgroup_storage_free(storage[stype]); =20 return ret; } --=20 2.17.1