From mboxrd@z Thu Jan 1 00:00:00 1970 Return-Path: X-Spam-Checker-Version: SpamAssassin 3.4.0 (2014-02-07) on aws-us-west-2-korg-lkml-1.web.codeaurora.org X-Spam-Level: X-Spam-Status: No, score=-0.8 required=3.0 tests=HEADER_FROM_DIFFERENT_DOMAINS, MAILING_LIST_MULTI,SPF_PASS autolearn=ham autolearn_force=no version=3.4.0 Received: from mail.kernel.org (mail.kernel.org [198.145.29.99]) by smtp.lore.kernel.org (Postfix) with ESMTP id B7062C433F5 for ; Fri, 7 Sep 2018 03:29:46 +0000 (UTC) Received: from vger.kernel.org (vger.kernel.org [209.132.180.67]) by mail.kernel.org (Postfix) with ESMTP id 61624205F4 for ; Fri, 7 Sep 2018 03:29:46 +0000 (UTC) DMARC-Filter: OpenDMARC Filter v1.3.2 mail.kernel.org 61624205F4 Authentication-Results: mail.kernel.org; dmarc=fail (p=none dis=none) header.from=redhat.com Authentication-Results: mail.kernel.org; spf=none smtp.mailfrom=linux-kernel-owner@vger.kernel.org Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1727413AbeIGII3 (ORCPT ); Fri, 7 Sep 2018 04:08:29 -0400 Received: from mx3-rdu2.redhat.com ([66.187.233.73]:38534 "EHLO mx1.redhat.com" rhost-flags-OK-OK-OK-FAIL) by vger.kernel.org with ESMTP id S1725999AbeIGII3 (ORCPT ); Fri, 7 Sep 2018 04:08:29 -0400 Received: from smtp.corp.redhat.com (int-mx03.intmail.prod.int.rdu2.redhat.com [10.11.54.3]) (using TLSv1.2 with cipher AECDH-AES256-SHA (256/256 bits)) (No client certificate requested) by mx1.redhat.com (Postfix) with ESMTPS id C0283401EF3D; Fri, 7 Sep 2018 03:29:42 +0000 (UTC) Received: from [10.72.12.125] (ovpn-12-125.pek2.redhat.com [10.72.12.125]) by smtp.corp.redhat.com (Postfix) with ESMTPS id 622C010F1BFF; Fri, 7 Sep 2018 03:29:37 +0000 (UTC) Subject: Re: [PATCH net-next 06/11] tuntap: split out XDP logic To: "Michael S. Tsirkin" Cc: netdev@vger.kernel.org, linux-kernel@vger.kernel.org, kvm@vger.kernel.org, virtualization@lists.linux-foundation.org References: <20180906040526.22518-1-jasowang@redhat.com> <20180906040526.22518-7-jasowang@redhat.com> <20180906131703-mutt-send-email-mst@kernel.org> From: Jason Wang Message-ID: <84ae17c1-f34b-610d-a5a1-ba8dce1732f3@redhat.com> Date: Fri, 7 Sep 2018 11:29:34 +0800 User-Agent: Mozilla/5.0 (X11; Linux x86_64; rv:52.0) Gecko/20100101 Thunderbird/52.9.1 MIME-Version: 1.0 In-Reply-To: <20180906131703-mutt-send-email-mst@kernel.org> Content-Type: text/plain; charset=utf-8; format=flowed Content-Transfer-Encoding: 8bit Content-Language: en-US X-Scanned-By: MIMEDefang 2.78 on 10.11.54.3 X-Greylist: Sender IP whitelisted, not delayed by milter-greylist-4.5.16 (mx1.redhat.com [10.11.55.5]); Fri, 07 Sep 2018 03:29:42 +0000 (UTC) X-Greylist: inspected by milter-greylist-4.5.16 (mx1.redhat.com [10.11.55.5]); Fri, 07 Sep 2018 03:29:42 +0000 (UTC) for IP:'10.11.54.3' DOMAIN:'int-mx03.intmail.prod.int.rdu2.redhat.com' HELO:'smtp.corp.redhat.com' FROM:'jasowang@redhat.com' RCPT:'' Sender: linux-kernel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org On 2018年09月07日 01:21, Michael S. Tsirkin wrote: > On Thu, Sep 06, 2018 at 12:05:21PM +0800, Jason Wang wrote: >> This patch split out XDP logic into a single function. This make it to >> be reused by XDP batching path in the following patch. >> >> Signed-off-by: Jason Wang >> --- >> drivers/net/tun.c | 84 ++++++++++++++++++++++++++++------------------- >> 1 file changed, 51 insertions(+), 33 deletions(-) >> >> diff --git a/drivers/net/tun.c b/drivers/net/tun.c >> index 389aa0727cc6..21b125020b3b 100644 >> --- a/drivers/net/tun.c >> +++ b/drivers/net/tun.c >> @@ -1635,6 +1635,44 @@ static bool tun_can_build_skb(struct tun_struct *tun, struct tun_file *tfile, >> return true; >> } >> >> +static u32 tun_do_xdp(struct tun_struct *tun, >> + struct tun_file *tfile, >> + struct bpf_prog *xdp_prog, >> + struct xdp_buff *xdp, >> + int *err) >> +{ >> + u32 act = bpf_prog_run_xdp(xdp_prog, xdp); >> + >> + switch (act) { >> + case XDP_REDIRECT: >> + *err = xdp_do_redirect(tun->dev, xdp, xdp_prog); >> + xdp_do_flush_map(); >> + if (*err) >> + break; >> + goto out; >> + case XDP_TX: >> + *err = tun_xdp_tx(tun->dev, xdp); >> + if (*err < 0) >> + break; >> + *err = 0; >> + goto out; >> + case XDP_PASS: >> + goto out; > Do we need goto? why not just return? I don't see any difference. > >> + default: >> + bpf_warn_invalid_xdp_action(act); >> + /* fall through */ >> + case XDP_ABORTED: >> + trace_xdp_exception(tun->dev, xdp_prog, act); >> + /* fall through */ >> + case XDP_DROP: >> + break; >> + } >> + >> + put_page(virt_to_head_page(xdp->data_hard_start)); > put here because caller does get_page :( Not pretty. > I'd move this out to the caller. Then we need a switch in the caller, not sure it's better. > >> +out: >> + return act; > How about combining err and act? err is < 0 XDP_PASS is > 0. > No need for pointers then. Ok. > >> +} >> + >> static struct sk_buff *tun_build_skb(struct tun_struct *tun, >> struct tun_file *tfile, >> struct iov_iter *from, >> @@ -1645,10 +1683,10 @@ static struct sk_buff *tun_build_skb(struct tun_struct *tun, >> struct sk_buff *skb = NULL; >> struct bpf_prog *xdp_prog; >> int buflen = SKB_DATA_ALIGN(sizeof(struct skb_shared_info)); >> - unsigned int delta = 0; >> char *buf; >> size_t copied; >> - int err, pad = TUN_RX_PAD; >> + int pad = TUN_RX_PAD; >> + int err = 0; >> >> rcu_read_lock(); >> xdp_prog = rcu_dereference(tun->xdp_prog); >> @@ -1685,9 +1723,8 @@ static struct sk_buff *tun_build_skb(struct tun_struct *tun, >> local_bh_disable(); >> rcu_read_lock(); >> xdp_prog = rcu_dereference(tun->xdp_prog); >> - if (xdp_prog && !*skb_xdp) { >> + if (xdp_prog) { >> struct xdp_buff xdp; >> - void *orig_data; >> u32 act; >> >> xdp.data_hard_start = buf; >> @@ -1695,33 +1732,14 @@ static struct sk_buff *tun_build_skb(struct tun_struct *tun, >> xdp_set_data_meta_invalid(&xdp); >> xdp.data_end = xdp.data + len; >> xdp.rxq = &tfile->xdp_rxq; >> - orig_data = xdp.data; >> - act = bpf_prog_run_xdp(xdp_prog, &xdp); >> - >> - switch (act) { >> - case XDP_REDIRECT: >> - err = xdp_do_redirect(tun->dev, &xdp, xdp_prog); >> - xdp_do_flush_map(); >> - if (err) >> - goto err_xdp; >> - goto out; >> - case XDP_TX: >> - if (tun_xdp_tx(tun->dev, &xdp) < 0) >> - goto err_xdp; >> - goto out; >> - case XDP_PASS: >> - delta = orig_data - xdp.data; >> - len = xdp.data_end - xdp.data; >> - break; >> - default: >> - bpf_warn_invalid_xdp_action(act); >> - /* fall through */ >> - case XDP_ABORTED: >> - trace_xdp_exception(tun->dev, xdp_prog, act); >> - /* fall through */ >> - case XDP_DROP: >> + act = tun_do_xdp(tun, tfile, xdp_prog, &xdp, &err); >> + if (err) >> goto err_xdp; >> - } >> + if (act != XDP_PASS) >> + goto out; > likely? It depends on the XDP program, so I tend not to use it. > >> + >> + pad = xdp.data - xdp.data_hard_start; >> + len = xdp.data_end - xdp.data; >> } >> rcu_read_unlock(); >> local_bh_enable(); >> @@ -1729,18 +1747,18 @@ static struct sk_buff *tun_build_skb(struct tun_struct *tun, >> build: >> skb = build_skb(buf, buflen); >> if (!skb) { >> + put_page(alloc_frag->page); >> skb = ERR_PTR(-ENOMEM); >> goto out; >> } >> >> - skb_reserve(skb, pad - delta); >> + skb_reserve(skb, pad); >> skb_put(skb, len); >> >> return skb; >> >> err_xdp: >> - alloc_frag->offset -= buflen; >> - put_page(alloc_frag->page); >> + this_cpu_inc(tun->pcpu_stats->rx_dropped); > > This fixes bug in previous patch which dropped it. OK :) Yes, but let me move this to the buggy patch. Thanks >> out: >> rcu_read_unlock(); >> local_bh_enable(); >> -- >> 2.17.1 From mboxrd@z Thu Jan 1 00:00:00 1970 From: Jason Wang Subject: Re: [PATCH net-next 06/11] tuntap: split out XDP logic Date: Fri, 7 Sep 2018 11:29:34 +0800 Message-ID: <84ae17c1-f34b-610d-a5a1-ba8dce1732f3@redhat.com> References: <20180906040526.22518-1-jasowang@redhat.com> <20180906040526.22518-7-jasowang@redhat.com> <20180906131703-mutt-send-email-mst@kernel.org> Mime-Version: 1.0 Content-Type: text/plain; charset="utf-8"; Format="flowed" Content-Transfer-Encoding: base64 Cc: netdev@vger.kernel.org, linux-kernel@vger.kernel.org, kvm@vger.kernel.org, virtualization@lists.linux-foundation.org To: "Michael S. Tsirkin" Return-path: In-Reply-To: <20180906131703-mutt-send-email-mst@kernel.org> Content-Language: en-US List-Unsubscribe: , List-Archive: List-Post: List-Help: List-Subscribe: , Sender: virtualization-bounces@lists.linux-foundation.org Errors-To: virtualization-bounces@lists.linux-foundation.org List-Id: netdev.vger.kernel.org CgpPbiAyMDE45bm0MDnmnIgwN+aXpSAwMToyMSwgTWljaGFlbCBTLiBUc2lya2luIHdyb3RlOgo+ IE9uIFRodSwgU2VwIDA2LCAyMDE4IGF0IDEyOjA1OjIxUE0gKzA4MDAsIEphc29uIFdhbmcgd3Jv dGU6Cj4+IFRoaXMgcGF0Y2ggc3BsaXQgb3V0IFhEUCBsb2dpYyBpbnRvIGEgc2luZ2xlIGZ1bmN0 aW9uLiBUaGlzIG1ha2UgaXQgdG8KPj4gYmUgcmV1c2VkIGJ5IFhEUCBiYXRjaGluZyBwYXRoIGlu IHRoZSBmb2xsb3dpbmcgcGF0Y2guCj4+Cj4+IFNpZ25lZC1vZmYtYnk6IEphc29uIFdhbmcgPGph c293YW5nQHJlZGhhdC5jb20+Cj4+IC0tLQo+PiAgIGRyaXZlcnMvbmV0L3R1bi5jIHwgODQgKysr KysrKysrKysrKysrKysrKysrKysrKysrKy0tLS0tLS0tLS0tLS0tLS0tLS0KPj4gICAxIGZpbGUg Y2hhbmdlZCwgNTEgaW5zZXJ0aW9ucygrKSwgMzMgZGVsZXRpb25zKC0pCj4+Cj4+IGRpZmYgLS1n aXQgYS9kcml2ZXJzL25ldC90dW4uYyBiL2RyaXZlcnMvbmV0L3R1bi5jCj4+IGluZGV4IDM4OWFh MDcyN2NjNi4uMjFiMTI1MDIwYjNiIDEwMDY0NAo+PiAtLS0gYS9kcml2ZXJzL25ldC90dW4uYwo+ PiArKysgYi9kcml2ZXJzL25ldC90dW4uYwo+PiBAQCAtMTYzNSw2ICsxNjM1LDQ0IEBAIHN0YXRp YyBib29sIHR1bl9jYW5fYnVpbGRfc2tiKHN0cnVjdCB0dW5fc3RydWN0ICp0dW4sIHN0cnVjdCB0 dW5fZmlsZSAqdGZpbGUsCj4+ICAgCXJldHVybiB0cnVlOwo+PiAgIH0KPj4gICAKPj4gK3N0YXRp YyB1MzIgdHVuX2RvX3hkcChzdHJ1Y3QgdHVuX3N0cnVjdCAqdHVuLAo+PiArCQkgICAgICBzdHJ1 Y3QgdHVuX2ZpbGUgKnRmaWxlLAo+PiArCQkgICAgICBzdHJ1Y3QgYnBmX3Byb2cgKnhkcF9wcm9n LAo+PiArCQkgICAgICBzdHJ1Y3QgeGRwX2J1ZmYgKnhkcCwKPj4gKwkJICAgICAgaW50ICplcnIp Cj4+ICt7Cj4+ICsJdTMyIGFjdCA9IGJwZl9wcm9nX3J1bl94ZHAoeGRwX3Byb2csIHhkcCk7Cj4+ ICsKPj4gKwlzd2l0Y2ggKGFjdCkgewo+PiArCWNhc2UgWERQX1JFRElSRUNUOgo+PiArCQkqZXJy ID0geGRwX2RvX3JlZGlyZWN0KHR1bi0+ZGV2LCB4ZHAsIHhkcF9wcm9nKTsKPj4gKwkJeGRwX2Rv X2ZsdXNoX21hcCgpOwo+PiArCQlpZiAoKmVycikKPj4gKwkJCWJyZWFrOwo+PiArCQlnb3RvIG91 dDsKPj4gKwljYXNlIFhEUF9UWDoKPj4gKwkJKmVyciA9IHR1bl94ZHBfdHgodHVuLT5kZXYsIHhk cCk7Cj4+ICsJCWlmICgqZXJyIDwgMCkKPj4gKwkJCWJyZWFrOwo+PiArCQkqZXJyID0gMDsKPj4g KwkJZ290byBvdXQ7Cj4+ICsJY2FzZSBYRFBfUEFTUzoKPj4gKwkJZ290byBvdXQ7Cj4gRG8gd2Ug bmVlZCBnb3RvPyB3aHkgbm90IGp1c3QgcmV0dXJuPwoKSSBkb24ndCBzZWUgYW55IGRpZmZlcmVu Y2UuCgo+Cj4+ICsJZGVmYXVsdDoKPj4gKwkJYnBmX3dhcm5faW52YWxpZF94ZHBfYWN0aW9uKGFj dCk7Cj4+ICsJCS8qIGZhbGwgdGhyb3VnaCAqLwo+PiArCWNhc2UgWERQX0FCT1JURUQ6Cj4+ICsJ CXRyYWNlX3hkcF9leGNlcHRpb24odHVuLT5kZXYsIHhkcF9wcm9nLCBhY3QpOwo+PiArCQkvKiBm YWxsIHRocm91Z2ggKi8KPj4gKwljYXNlIFhEUF9EUk9QOgo+PiArCQlicmVhazsKPj4gKwl9Cj4+ ICsKPj4gKwlwdXRfcGFnZSh2aXJ0X3RvX2hlYWRfcGFnZSh4ZHAtPmRhdGFfaGFyZF9zdGFydCkp Owo+IHB1dCBoZXJlIGJlY2F1c2UgY2FsbGVyIGRvZXMgZ2V0X3BhZ2UgOiggTm90IHByZXR0eS4K PiBJJ2QgbW92ZSB0aGlzIG91dCB0byB0aGUgY2FsbGVyLgoKVGhlbiB3ZSBuZWVkIGEgc3dpdGNo IGluIHRoZSBjYWxsZXIsIG5vdCBzdXJlIGl0J3MgYmV0dGVyLgoKPgo+PiArb3V0Ogo+PiArCXJl dHVybiBhY3Q7Cj4gSG93IGFib3V0IGNvbWJpbmluZyBlcnIgYW5kIGFjdD8gZXJyIGlzIDwgMCBY RFBfUEFTUyBpcyA+IDAuCj4gTm8gbmVlZCBmb3IgcG9pbnRlcnMgdGhlbi4KCk9rLgoKPgo+PiAr fQo+PiArCj4+ICAgc3RhdGljIHN0cnVjdCBza19idWZmICp0dW5fYnVpbGRfc2tiKHN0cnVjdCB0 dW5fc3RydWN0ICp0dW4sCj4+ICAgCQkJCSAgICAgc3RydWN0IHR1bl9maWxlICp0ZmlsZSwKPj4g ICAJCQkJICAgICBzdHJ1Y3QgaW92X2l0ZXIgKmZyb20sCj4+IEBAIC0xNjQ1LDEwICsxNjgzLDEw IEBAIHN0YXRpYyBzdHJ1Y3Qgc2tfYnVmZiAqdHVuX2J1aWxkX3NrYihzdHJ1Y3QgdHVuX3N0cnVj dCAqdHVuLAo+PiAgIAlzdHJ1Y3Qgc2tfYnVmZiAqc2tiID0gTlVMTDsKPj4gICAJc3RydWN0IGJw Zl9wcm9nICp4ZHBfcHJvZzsKPj4gICAJaW50IGJ1ZmxlbiA9IFNLQl9EQVRBX0FMSUdOKHNpemVv ZihzdHJ1Y3Qgc2tiX3NoYXJlZF9pbmZvKSk7Cj4+IC0JdW5zaWduZWQgaW50IGRlbHRhID0gMDsK Pj4gICAJY2hhciAqYnVmOwo+PiAgIAlzaXplX3QgY29waWVkOwo+PiAtCWludCBlcnIsIHBhZCA9 IFRVTl9SWF9QQUQ7Cj4+ICsJaW50IHBhZCA9IFRVTl9SWF9QQUQ7Cj4+ICsJaW50IGVyciA9IDA7 Cj4+ICAgCj4+ICAgCXJjdV9yZWFkX2xvY2soKTsKPj4gICAJeGRwX3Byb2cgPSByY3VfZGVyZWZl cmVuY2UodHVuLT54ZHBfcHJvZyk7Cj4+IEBAIC0xNjg1LDkgKzE3MjMsOCBAQCBzdGF0aWMgc3Ry dWN0IHNrX2J1ZmYgKnR1bl9idWlsZF9za2Ioc3RydWN0IHR1bl9zdHJ1Y3QgKnR1biwKPj4gICAJ bG9jYWxfYmhfZGlzYWJsZSgpOwo+PiAgIAlyY3VfcmVhZF9sb2NrKCk7Cj4+ICAgCXhkcF9wcm9n ID0gcmN1X2RlcmVmZXJlbmNlKHR1bi0+eGRwX3Byb2cpOwo+PiAtCWlmICh4ZHBfcHJvZyAmJiAh KnNrYl94ZHApIHsKPj4gKwlpZiAoeGRwX3Byb2cpIHsKPj4gICAJCXN0cnVjdCB4ZHBfYnVmZiB4 ZHA7Cj4+IC0JCXZvaWQgKm9yaWdfZGF0YTsKPj4gICAJCXUzMiBhY3Q7Cj4+ICAgCj4+ICAgCQl4 ZHAuZGF0YV9oYXJkX3N0YXJ0ID0gYnVmOwo+PiBAQCAtMTY5NSwzMyArMTczMiwxNCBAQCBzdGF0 aWMgc3RydWN0IHNrX2J1ZmYgKnR1bl9idWlsZF9za2Ioc3RydWN0IHR1bl9zdHJ1Y3QgKnR1biwK Pj4gICAJCXhkcF9zZXRfZGF0YV9tZXRhX2ludmFsaWQoJnhkcCk7Cj4+ICAgCQl4ZHAuZGF0YV9l bmQgPSB4ZHAuZGF0YSArIGxlbjsKPj4gICAJCXhkcC5yeHEgPSAmdGZpbGUtPnhkcF9yeHE7Cj4+ IC0JCW9yaWdfZGF0YSA9IHhkcC5kYXRhOwo+PiAtCQlhY3QgPSBicGZfcHJvZ19ydW5feGRwKHhk cF9wcm9nLCAmeGRwKTsKPj4gLQo+PiAtCQlzd2l0Y2ggKGFjdCkgewo+PiAtCQljYXNlIFhEUF9S RURJUkVDVDoKPj4gLQkJCWVyciA9IHhkcF9kb19yZWRpcmVjdCh0dW4tPmRldiwgJnhkcCwgeGRw X3Byb2cpOwo+PiAtCQkJeGRwX2RvX2ZsdXNoX21hcCgpOwo+PiAtCQkJaWYgKGVycikKPj4gLQkJ CQlnb3RvIGVycl94ZHA7Cj4+IC0JCQlnb3RvIG91dDsKPj4gLQkJY2FzZSBYRFBfVFg6Cj4+IC0J CQlpZiAodHVuX3hkcF90eCh0dW4tPmRldiwgJnhkcCkgPCAwKQo+PiAtCQkJCWdvdG8gZXJyX3hk cDsKPj4gLQkJCWdvdG8gb3V0Owo+PiAtCQljYXNlIFhEUF9QQVNTOgo+PiAtCQkJZGVsdGEgPSBv cmlnX2RhdGEgLSB4ZHAuZGF0YTsKPj4gLQkJCWxlbiA9IHhkcC5kYXRhX2VuZCAtIHhkcC5kYXRh Owo+PiAtCQkJYnJlYWs7Cj4+IC0JCWRlZmF1bHQ6Cj4+IC0JCQlicGZfd2Fybl9pbnZhbGlkX3hk cF9hY3Rpb24oYWN0KTsKPj4gLQkJCS8qIGZhbGwgdGhyb3VnaCAqLwo+PiAtCQljYXNlIFhEUF9B Qk9SVEVEOgo+PiAtCQkJdHJhY2VfeGRwX2V4Y2VwdGlvbih0dW4tPmRldiwgeGRwX3Byb2csIGFj dCk7Cj4+IC0JCQkvKiBmYWxsIHRocm91Z2ggKi8KPj4gLQkJY2FzZSBYRFBfRFJPUDoKPj4gKwkJ YWN0ID0gdHVuX2RvX3hkcCh0dW4sIHRmaWxlLCB4ZHBfcHJvZywgJnhkcCwgJmVycik7Cj4+ICsJ CWlmIChlcnIpCj4+ICAgCQkJZ290byBlcnJfeGRwOwo+PiAtCQl9Cj4+ICsJCWlmIChhY3QgIT0g WERQX1BBU1MpCj4+ICsJCQlnb3RvIG91dDsKPiBsaWtlbHk/CgpJdCBkZXBlbmRzIG9uIHRoZSBY RFAgcHJvZ3JhbSwgc28gSSB0ZW5kIG5vdCB0byB1c2UgaXQuCgo+Cj4+ICsKPj4gKwkJcGFkID0g eGRwLmRhdGEgLSB4ZHAuZGF0YV9oYXJkX3N0YXJ0Owo+PiArCQlsZW4gPSB4ZHAuZGF0YV9lbmQg LSB4ZHAuZGF0YTsKPj4gICAJfQo+PiAgIAlyY3VfcmVhZF91bmxvY2soKTsKPj4gICAJbG9jYWxf YmhfZW5hYmxlKCk7Cj4+IEBAIC0xNzI5LDE4ICsxNzQ3LDE4IEBAIHN0YXRpYyBzdHJ1Y3Qgc2tf YnVmZiAqdHVuX2J1aWxkX3NrYihzdHJ1Y3QgdHVuX3N0cnVjdCAqdHVuLAo+PiAgIGJ1aWxkOgo+ PiAgIAlza2IgPSBidWlsZF9za2IoYnVmLCBidWZsZW4pOwo+PiAgIAlpZiAoIXNrYikgewo+PiAr CQlwdXRfcGFnZShhbGxvY19mcmFnLT5wYWdlKTsKPj4gICAJCXNrYiA9IEVSUl9QVFIoLUVOT01F TSk7Cj4+ICAgCQlnb3RvIG91dDsKPj4gICAJfQo+PiAgIAo+PiAtCXNrYl9yZXNlcnZlKHNrYiwg cGFkIC0gZGVsdGEpOwo+PiArCXNrYl9yZXNlcnZlKHNrYiwgcGFkKTsKPj4gICAJc2tiX3B1dChz a2IsIGxlbik7Cj4+ICAgCj4+ICAgCXJldHVybiBza2I7Cj4+ICAgCj4+ICAgZXJyX3hkcDoKPj4g LQlhbGxvY19mcmFnLT5vZmZzZXQgLT0gYnVmbGVuOwo+PiAtCXB1dF9wYWdlKGFsbG9jX2ZyYWct PnBhZ2UpOwo+PiArCXRoaXNfY3B1X2luYyh0dW4tPnBjcHVfc3RhdHMtPnJ4X2Ryb3BwZWQpOwo+ Cj4gVGhpcyBmaXhlcyBidWcgaW4gcHJldmlvdXMgcGF0Y2ggd2hpY2ggZHJvcHBlZCBpdC4gT0sg OikKClllcywgYnV0IGxldCBtZSBtb3ZlIHRoaXMgdG8gdGhlIGJ1Z2d5IHBhdGNoLgoKVGhhbmtz Cgo+PiAgIG91dDoKPj4gICAJcmN1X3JlYWRfdW5sb2NrKCk7Cj4+ICAgCWxvY2FsX2JoX2VuYWJs ZSgpOwo+PiAtLSAKPj4gMi4xNy4xCgpfX19fX19fX19fX19fX19fX19fX19fX19fX19fX19fX19f X19fX19fX19fX19fXwpWaXJ0dWFsaXphdGlvbiBtYWlsaW5nIGxpc3QKVmlydHVhbGl6YXRpb25A bGlzdHMubGludXgtZm91bmRhdGlvbi5vcmcKaHR0cHM6Ly9saXN0cy5saW51eGZvdW5kYXRpb24u b3JnL21haWxtYW4vbGlzdGluZm8vdmlydHVhbGl6YXRpb24=