Received: by 2002:a05:6a10:f347:0:0:0:0 with SMTP id d7csp1147753pxu; Fri, 27 Nov 2020 00:25:54 -0800 (PST) X-Google-Smtp-Source: ABdhPJxXW9gMScfvYb/J2SmRB2vaIPf21wDQgyPbYhqJdAVdyqo6Or3Ufc9YLnFwN6EnjRqOr43b X-Received: by 2002:a05:6402:1c0a:: with SMTP id ck10mr6336416edb.266.1606465553765; Fri, 27 Nov 2020 00:25:53 -0800 (PST) ARC-Seal: i=1; a=rsa-sha256; t=1606465553; cv=none; d=google.com; s=arc-20160816; b=sbivFnbkSwLgFCUVyXUN9M54J9KRh53m25Bq1PQ31XmHW4MQ0KphCX0P5n5Y4Tfilr Iir/7/757kJ7k7vLbB4uVGVJf9vnX5TWmXKjMT9Gks49FO75tGm3++Ua0KH+Wet/dGWM 4x4n3MsQEuUKTaAmOq7Ya6LHHoW8ChgVGrn6R2sWt04PA9Hnwx1bqzwoE9ZmC+Ey0BlV LwPq+9PTNDV7s7PPkLjmtPI4RleyMiIosTMJ4bKLYIbwMY9HxVrJyOZa2XT1p0RROAZi sv3G3VhBKAl1ydFfEnUpWf5x0MGzLfUKXJFDOFvZB3FzYUpFaw+5CoR6GMfWeU9c07oJ b84A== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:in-reply-to:mime-version:user-agent:date :message-id:subject:from:references:cc:to; bh=SLXrrOD2lwcOUipmcAZb4AqSYwt2pa/Rwsugf+zHIjE=; b=oSD98gemP6RCc9Ld/EP7aurX2gwqPjlGmaDyAnYAKxZYiVtlu17887i8/acqG1zd4A s2redUUHdl8jHFRlhQJChjDcnDRNkKXeHI0sUBzPNCWE9oJtnr+nnq2DFF8D+YSPR3kO QzgyVzQDPJz1FeRd705V2imZJVivXl60s5cJ2HNu2z96tEBfYJedUQxdX/2Ps/C4kFl5 cU8eqF/vQKmuYswMj9c3fvWoQ7eSw2oEbF6z2att9blr9G4GzFO6aP5I/l7jI5a+IxJY Ml2M2DX+rnVwV2vvK29Dz3B7Fl774cnnpEgSw6bkY2MhTOcnrXP89ix+aazjzl18KO1F HkWw== ARC-Authentication-Results: i=1; mx.google.com; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.18 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org Return-Path: Received: from vger.kernel.org (vger.kernel.org. [23.128.96.18]) by mx.google.com with ESMTP id p3si4830050edp.84.2020.11.27.00.25.30; Fri, 27 Nov 2020 00:25:53 -0800 (PST) Received-SPF: pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.18 as permitted sender) client-ip=23.128.96.18; Authentication-Results: mx.google.com; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.18 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S2389783AbgKZNFy (ORCPT + 99 others); Thu, 26 Nov 2020 08:05:54 -0500 Received: from mx2.suse.de ([195.135.220.15]:49110 "EHLO mx2.suse.de" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S2389769AbgKZNFy (ORCPT ); Thu, 26 Nov 2020 08:05:54 -0500 X-Virus-Scanned: by amavisd-new at test-mx.suse.de Received: from relay2.suse.de (unknown [195.135.221.27]) by mx2.suse.de (Postfix) with ESMTP id 66C4BAD21; Thu, 26 Nov 2020 13:05:51 +0000 (UTC) To: Tian Tao , airlied@linux.ie, daniel@ffwll.ch, kraxel@redhat.com, alexander.deucher@amd.com, tglx@linutronix.de, dri-devel@lists.freedesktop.org, xinliang.liu@linaro.org, maarten.lankhorst@linux.intel.com, mripard@kernel.org Cc: linux-kernel@vger.kernel.org References: <1606392140-57954-1-git-send-email-tiantao6@hisilicon.com> <1606392140-57954-2-git-send-email-tiantao6@hisilicon.com> From: Thomas Zimmermann Subject: Re: [PATCH drm/hisilicon 1/3] drm/hisilicon: Code refactoring for hibmc_drm_drv Message-ID: Date: Thu, 26 Nov 2020 14:05:49 +0100 User-Agent: Mozilla/5.0 (X11; Linux x86_64; rv:78.0) Gecko/20100101 Thunderbird/78.4.3 MIME-Version: 1.0 In-Reply-To: <1606392140-57954-2-git-send-email-tiantao6@hisilicon.com> Content-Type: multipart/signed; micalg=pgp-sha256; protocol="application/pgp-signature"; boundary="IXOfhhr90WUTexsH77RH73TvXPY0rJsGv" Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org This is an OpenPGP/MIME signed message (RFC 4880 and 3156) --IXOfhhr90WUTexsH77RH73TvXPY0rJsGv Content-Type: multipart/mixed; boundary="TtE0nB4yElvNRRDtk7uZnjah9k8KfGcy8"; protected-headers="v1" From: Thomas Zimmermann To: Tian Tao , airlied@linux.ie, daniel@ffwll.ch, kraxel@redhat.com, alexander.deucher@amd.com, tglx@linutronix.de, dri-devel@lists.freedesktop.org, xinliang.liu@linaro.org, maarten.lankhorst@linux.intel.com, mripard@kernel.org Cc: linux-kernel@vger.kernel.org Message-ID: Subject: Re: [PATCH drm/hisilicon 1/3] drm/hisilicon: Code refactoring for hibmc_drm_drv References: <1606392140-57954-1-git-send-email-tiantao6@hisilicon.com> <1606392140-57954-2-git-send-email-tiantao6@hisilicon.com> In-Reply-To: <1606392140-57954-2-git-send-email-tiantao6@hisilicon.com> --TtE0nB4yElvNRRDtk7uZnjah9k8KfGcy8 Content-Type: multipart/mixed; boundary="------------23E1404708D2B7816DE29839" Content-Language: en-US This is a multi-part message in MIME format. --------------23E1404708D2B7816DE29839 Content-Type: text/plain; charset=utf-8; format=flowed Content-Transfer-Encoding: quoted-printable Hi Am 26.11.20 um 13:02 schrieb Tian Tao: > Use the devm_drm_dev_alloc provided by the drm framework to alloc > a struct hibmc_drm_private. >=20 > Signed-off-by: Tian Tao > --- > drivers/gpu/drm/hisilicon/hibmc/hibmc_drm_de.c | 2 +- > drivers/gpu/drm/hisilicon/hibmc/hibmc_drm_drv.c | 51 +++++++++++----= --------- > drivers/gpu/drm/hisilicon/hibmc/hibmc_drm_drv.h | 2 +- > drivers/gpu/drm/hisilicon/hibmc/hibmc_drm_vdac.c | 2 +- > drivers/gpu/drm/hisilicon/hibmc/hibmc_ttm.c | 8 ++-- > 5 files changed, 31 insertions(+), 34 deletions(-) >=20 > diff --git a/drivers/gpu/drm/hisilicon/hibmc/hibmc_drm_de.c b/drivers/g= pu/drm/hisilicon/hibmc/hibmc_drm_de.c > index ea962ac..096eea9 100644 > --- a/drivers/gpu/drm/hisilicon/hibmc/hibmc_drm_de.c > +++ b/drivers/gpu/drm/hisilicon/hibmc/hibmc_drm_de.c > @@ -499,7 +499,7 @@ static const struct drm_crtc_helper_funcs hibmc_crt= c_helper_funcs =3D { > =20 > int hibmc_de_init(struct hibmc_drm_private *priv) > { > - struct drm_device *dev =3D priv->dev; > + struct drm_device *dev =3D &priv->dev; > struct drm_crtc *crtc =3D &priv->crtc; > struct drm_plane *plane =3D &priv->primary_plane; > int ret; > diff --git a/drivers/gpu/drm/hisilicon/hibmc/hibmc_drm_drv.c b/drivers/= gpu/drm/hisilicon/hibmc/hibmc_drm_drv.c > index d845657..ea3d81b 100644 > --- a/drivers/gpu/drm/hisilicon/hibmc/hibmc_drm_drv.c > +++ b/drivers/gpu/drm/hisilicon/hibmc/hibmc_drm_drv.c > @@ -80,30 +80,31 @@ static const struct dev_pm_ops hibmc_pm_ops =3D { > static int hibmc_kms_init(struct hibmc_drm_private *priv) > { > int ret; > + struct drm_device *dev =3D &priv->dev; I think that's good style. Pointers deserve their own local variable if=20 they are used multiple times within a function. I would move this change = into a separate patch, because it's only semi-related to the actual chang= es. > =20 > - drm_mode_config_init(priv->dev); > + drm_mode_config_init(dev); > priv->mode_config_initialized =3D true; > =20 > - priv->dev->mode_config.min_width =3D 0; > - priv->dev->mode_config.min_height =3D 0; > - priv->dev->mode_config.max_width =3D 1920; > - priv->dev->mode_config.max_height =3D 1200; > + dev->mode_config.min_width =3D 0; > + dev->mode_config.min_height =3D 0; > + dev->mode_config.max_width =3D 1920; > + dev->mode_config.max_height =3D 1200; > =20 > - priv->dev->mode_config.fb_base =3D priv->fb_base; > - priv->dev->mode_config.preferred_depth =3D 32; > - priv->dev->mode_config.prefer_shadow =3D 1; > + dev->mode_config.fb_base =3D priv->fb_base; > + dev->mode_config.preferred_depth =3D 32; > + dev->mode_config.prefer_shadow =3D 1; > =20 > - priv->dev->mode_config.funcs =3D (void *)&hibmc_mode_funcs; > + dev->mode_config.funcs =3D (void *)&hibmc_mode_funcs; > =20 > ret =3D hibmc_de_init(priv); > if (ret) { > - drm_err(priv->dev, "failed to init de: %d\n", ret); > + drm_err(dev, "failed to init de: %d\n", ret); > return ret; > } > =20 > ret =3D hibmc_vdac_init(priv); > if (ret) { > - drm_err(priv->dev, "failed to init vdac: %d\n", ret); > + drm_err(dev, "failed to init vdac: %d\n", ret); > return ret; > } > =20 > @@ -113,7 +114,7 @@ static int hibmc_kms_init(struct hibmc_drm_private = *priv) > static void hibmc_kms_fini(struct hibmc_drm_private *priv) > { > if (priv->mode_config_initialized) { > - drm_mode_config_cleanup(priv->dev); > + drm_mode_config_cleanup(&priv->dev); > priv->mode_config_initialized =3D false; > } > } > @@ -202,7 +203,7 @@ static void hibmc_hw_config(struct hibmc_drm_privat= e *priv) > =20 > static int hibmc_hw_map(struct hibmc_drm_private *priv) > { > - struct drm_device *dev =3D priv->dev; > + struct drm_device *dev =3D &priv->dev; > struct pci_dev *pdev =3D dev->pdev; > resource_size_t addr, size, ioaddr, iosize; > =20 > @@ -258,17 +259,9 @@ static int hibmc_unload(struct drm_device *dev) > =20 > static int hibmc_load(struct drm_device *dev) > { > - struct hibmc_drm_private *priv; > + struct hibmc_drm_private *priv =3D dev->dev_private; Please use to_hibmc_drm_private() instead. dev_private is deprecated. > int ret; > =20 > - priv =3D drmm_kzalloc(dev, sizeof(*priv), GFP_KERNEL); > - if (!priv) { > - drm_err(dev, "no memory to allocate for hibmc_drm_private\n"); > - return -ENOMEM; > - } > - dev->dev_private =3D priv; > - priv->dev =3D dev; > - > ret =3D hibmc_hw_init(priv); > if (ret) > goto err; > @@ -310,6 +303,7 @@ static int hibmc_load(struct drm_device *dev) > static int hibmc_pci_probe(struct pci_dev *pdev, > const struct pci_device_id *ent) > { > + struct hibmc_drm_private *priv; > struct drm_device *dev; > int ret; > =20 > @@ -318,19 +312,22 @@ static int hibmc_pci_probe(struct pci_dev *pdev, > if (ret) > return ret; > =20 > - dev =3D drm_dev_alloc(&hibmc_driver, &pdev->dev); > - if (IS_ERR(dev)) { > + priv =3D devm_drm_dev_alloc(&pdev->dev, &hibmc_driver, > + struct hibmc_drm_private, dev); > + if (IS_ERR(priv)) { > DRM_ERROR("failed to allocate drm_device\n"); > - return PTR_ERR(dev); > + return PTR_ERR(priv); > } > =20 > + dev =3D &priv->dev; > + dev->dev_private =3D priv; Don't do this assignment. After this patch, no code in hibmc should=20 touch dev_private. Instead, change to_hibmc_drm_private() to upcast from = dev. As in [1]. > dev->pdev =3D pdev; > pci_set_drvdata(pdev, dev); > =20 > ret =3D pci_enable_device(pdev); > if (ret) { > drm_err(dev, "failed to enable pci device: %d\n", ret); > - goto err_free; > + return ret; > } > =20 > ret =3D hibmc_load(dev); > @@ -354,8 +351,6 @@ static int hibmc_pci_probe(struct pci_dev *pdev, > hibmc_unload(dev); > err_disable: > pci_disable_device(pdev); > -err_free: > - drm_dev_put(dev); > =20 > return ret; > } > diff --git a/drivers/gpu/drm/hisilicon/hibmc/hibmc_drm_drv.h b/drivers/= gpu/drm/hisilicon/hibmc/hibmc_drm_drv.h > index f310a83..e35353a 100644 > --- a/drivers/gpu/drm/hisilicon/hibmc/hibmc_drm_drv.h > +++ b/drivers/gpu/drm/hisilicon/hibmc/hibmc_drm_drv.h > @@ -37,7 +37,7 @@ struct hibmc_drm_private { > resource_size_t fb_size; > =20 > /* drm */ > - struct drm_device *dev; > + struct drm_device dev; > struct drm_plane primary_plane; > struct drm_crtc crtc; > struct drm_encoder encoder; > diff --git a/drivers/gpu/drm/hisilicon/hibmc/hibmc_drm_vdac.c b/drivers= /gpu/drm/hisilicon/hibmc/hibmc_drm_vdac.c > index 74e26c2..d35548d 100644 > --- a/drivers/gpu/drm/hisilicon/hibmc/hibmc_drm_vdac.c > +++ b/drivers/gpu/drm/hisilicon/hibmc/hibmc_drm_vdac.c > @@ -96,7 +96,7 @@ static const struct drm_encoder_funcs hibmc_encoder_f= uncs =3D { > =20 > int hibmc_vdac_init(struct hibmc_drm_private *priv) > { > - struct drm_device *dev =3D priv->dev; > + struct drm_device *dev =3D &priv->dev; > struct hibmc_connector *hibmc_connector =3D &priv->connector; > struct drm_encoder *encoder =3D &priv->encoder; > struct drm_connector *connector =3D &hibmc_connector->base; > diff --git a/drivers/gpu/drm/hisilicon/hibmc/hibmc_ttm.c b/drivers/gpu/= drm/hisilicon/hibmc/hibmc_ttm.c > index 602ece1..e84fb81 100644 > --- a/drivers/gpu/drm/hisilicon/hibmc/hibmc_ttm.c > +++ b/drivers/gpu/drm/hisilicon/hibmc/hibmc_ttm.c > @@ -25,7 +25,7 @@ int hibmc_mm_init(struct hibmc_drm_private *hibmc) > { > struct drm_vram_mm *vmm; > int ret; > - struct drm_device *dev =3D hibmc->dev; > + struct drm_device *dev =3D &hibmc->dev; > =20 > vmm =3D drm_vram_helper_alloc_mm(dev, > pci_resource_start(dev->pdev, 0), > @@ -41,10 +41,12 @@ int hibmc_mm_init(struct hibmc_drm_private *hibmc) > =20 > void hibmc_mm_fini(struct hibmc_drm_private *hibmc) > { > - if (!hibmc->dev->vram_mm) > + struct drm_device *dev =3D &hibmc->dev; > + > + if (!dev->vram_mm) > return; > =20 > - drm_vram_helper_release_mm(hibmc->dev); > + drm_vram_helper_release_mm(dev); That's OK for now. A good follow-up patchset is the conversion to=20 managed DRM helpers. We have these for modesetting, vram helpers and othe= rs. > } > =20 > int hibmc_dumb_create(struct drm_file *file, struct drm_device *dev, >=20 --=20 Thomas Zimmermann Graphics Driver Developer SUSE Software Solutions Germany GmbH Maxfeldstr. 5, 90409 N=C3=BCrnberg, Germany (HRB 36809, AG N=C3=BCrnberg) Gesch=C3=A4ftsf=C3=BChrer: Felix Imend=C3=B6rffer --------------23E1404708D2B7816DE29839 Content-Type: application/pgp-keys; name="OpenPGP_0x680DC11D530B7A23.asc" Content-Transfer-Encoding: quoted-printable Content-Disposition: attachment; filename="OpenPGP_0x680DC11D530B7A23.asc" -----BEGIN PGP PUBLIC KEY BLOCK----- xsBNBFs50uABCADEHPidWt974CaxBVbrIBwqcq/WURinJ3+2WlIrKWspiP83vfZKaXhFYsdgX= H47 fDVbPPj+d6tQrw5lPQCyqjwrCPYnq3WlIBnGPJ4/jreTL6V+qfKRDlGLWFjZcsrPJGE0BeB5B= bqP 5erN1qylK9i3gPoQjXGhpBpQYwRrEyQyjuvk+Ev0K1Jc5tVDeJAuau3TGNgah4YchdHm3bkPj= z9E ErV85RwvImQ1dptvx6s7xzwXTgGAsaYZsL8WCwDaTuqFa1d1jjlaxg6+tZsB9GluwvIhSezPg= nEm imZDkGnZRRSFiGP8yjqTjjWuf0bSj5rUnTGiyLyRZRNGcXmu6hjlABEBAAHNKFRob21hcyBaa= W1t ZXJtYW5uIDx0emltbWVybWFubkBzdXNlLmNvbT7CwI4EEwEIADgCGwMFCwkIBwIGFQoJCAsCB= BYC AwECHgECF4AWIQRyF/usjOnPY0ShaOVoDcEdUwt6IwUCXvxIWAAKCRBoDcEdUwt6I+aZB/9ih= Onf G4Lgf1L87cvoXh95/bnaJ6aQhP6/ZeRleuCXflnyDajlm3c9loQr0r2bQUi7JeYwUKbBab2QS= GJm DMRGlLMnmzWB8mHmZ6bHAu+2Sth8SraE42p6BB9d8dlYEID+dl/D/xUBeulfkck5rloGtYqDi= +1Q DfkEZJaxVSZ6FFkXuQi/G9qcI4iklN2nv02iQ7mZe8WYAysix6s/6vIobhirEBreclSNxXqis= p8n 91+v855JC11EgRdUXMRK81IAaCKXP8zLx3ixku7mvP9Om61yerHSbeU2HZbIggZYQlFh6llJm= zF1 CjCWgPTJyk4t4kMTcNOw5ykD47vU/KW+wl0EEBECAB0WIQQn6OOmnzvP/7ktjmoud6EwEfXTw= gUC WzodVwAKCRAud6EwEfXTwidvAKDkOADDHfI0QNXqAZcg6i1kOndAYACeLXHBwpjnumkPSyoab= IiL +he8r3zCwHMEEAEIAB0WIQQeXZghmQijlU7YzFiqUDvJrg9HpwUCWznxsQAKCRCqUDvJrg9Hp= 42f CADIvsZcAd04PDFclRltHr2huy6s7+ZZA6PgYlMblEBh4bJA+dNPBTvzpJ7FJv/bmHOa+phWy= Urj EpfFGuOKGuWAfzgVAEu52fMrW3/mm+O26z1AKIu8hiZ/x9OAe4AM71ZO2lZrV1/53ZdzWnRuO= 45N GQcotU8oeVfT9okAfmozmWMmIMq7Q0K6bV8W3qiD5XfDNxjr2caxc/9WX1bZPUo3n0H23MNaA= Tpy Oz732UtDh6sKUAB1RfzBBd/REbjHD7+quwJGAdRScyDRncX1vNb2+wihy0ipA69XY3bkhR5iD= u5r A9enuiMe6J1IBMI1PZh+vOufB/M6cd2D9RULIJaJwsBzBBABCAAdFiEEuyNtt7Ge78bIRx1op= /N8 GYw5MYEFAls6MrsACgkQp/N8GYw5MYEnLQf/dwqlDJVQL2q+i8FFaqTMAm0n9jLRV6pN8JxFH= j0g voyWUOnQuNdAFgtKd26ZhN8NkLoSMO8E19eBPfLoBIFK5yNNVmRHAZm07MzGbA0uNWINJhmdR= bZM RMh0nneXjcEU/IvUmd8TPFTAd24X2mbzHgcaHMLJSVx1ohd4alRJXHIqDobKmiVwekyPnInJn= zWw iuZUkIotTkQple1PT/dF3S+KtPXBL6ldQ4NkAeCjsz4wnzSa9+VKOxEhiHM0PMzXSbkCMP+4m= Xy9 RMplBw9Dm9hN2PSouBPifIrSodiiSWZYXOEkzLiBAB0frCKR63Dnx9kvjCD9Pz5wLd/70rjqI= cLA jgQTAQgAOAIbAwULCQgHAgYVCgkICwIEFgIDAQIeAQIXgBYhBHIX+6yM6c9jRKFo5WgNwR1TC= 3oj BQJftODHAAoJEGgNwR1TC3ojZSIIAIV3makffp4P4leU8JSLt0aTNewsOhy7VQzKUtlCw3PKD= 3l/ SuymZhQKgH+n6sijzFauZnZ+x0T+Oy+dDVZb3sNJuuMUDIHw18EO9daZBMcueaS54FGe73lAp= HUl 7nxyocCxoqIG8+fP+75itV/ls2TSh5rJvjLvHC8J3NqfGlJ/jlSKrQUnzFbXfE5KGWiKNAn+I= 1a2 EE0I7uLpYgkdb8hcjtV9Rxr2ja+GWOaSoqB29P5GUzipkWo4144Q16JBO6QP2R9y/1ZK9VqH2= 5T8 lTKocLAaHCEdpDqY5KI15as9tIxlI1Vh+eqhTh/gwEm1ykO1gmrQ1zvGLDMB1EE6El3NJ1Rob= 21h cyBaaW1tZXJtYW5uIDx0emltbWVybWFubkBzdXNlLmRlPsLAjgQTAQgAOAIbAwULCQgHAgYVC= gkI CwIEFgIDAQIeAQIXgBYhBHIX+6yM6c9jRKFo5WgNwR1TC3ojBQJe/EheAAoJEGgNwR1TC3ojq= RgI AIoegtTp1prPzfHgTAuEPd8v58ssHubwi9tg69a8IJ+iMNozhs4iUou84WOLBJNjSieXHZRa8= fJj //2/sTuABn38AQ9FcKX9/B49hrdCo6c0WHHKqlPrSTzuXNKYyOdmSFd/pDhBb2Bn5DTxxH5RP= m/N U/C9nUlwi7Y+FgBlDNa5h592wmJfv0cJAfvF56C+QL65jHFOFIW9xSaTOAxxMXHGJHXki6Iwa= aTg s7QQlKQcd5XvvED1bwLyQ7rq+MEZo5N7IygpQMM3qqGMlCnDdyQ3W95rd0HCWpfa0oVRCOwdu= SL3 5hG7ONqBpvBj8z5GjSbt4HLJGvpeT0k37qzRExrCXQQQEQIAHRYhBCfo46afO8//uS2Oai53o= TAR 9dPCBQJbOh1XAAoJEC53oTAR9dPC05AAoIy0HQ2DBDYugQ42P4HfyxfZTIvKAJ0fqNBcBFW9S= tbR DEP9cfpNVOv8YMLAcwQQAQgAHRYhBB5dmCGZCKOVTtjMWKpQO8muD0enBQJbOfGzAAoJEKpQO= 8mu D0enL0wIAM2NTeUDCofBAkbWHGTZopclefbh0xGPYQEfttNyalp0hn1CrVO7OsX5eTjRqgyOa= 1C5 OAsNghCM4PUmrfv5cZ9+sNn9bRM50uVW9IFRlq8wwBY4+7QejJ5gs7DW/0tZIMZ6iTGKK0WEO= 7gd 2K9hXadPBScTdIqXeWH82meiqElnEQL+K9UeGUBrku+1EQIOxwziKwTDlTvhyJ+xmEKj0uWRc= Ocl 27xLS9XOWPGXcNQBtlZhF8e/E1kFRt5CPP5UBdUCN8qydUadseXivSNDiYob9dyJSFt7G0Bq4= /ac Ret5ANtGRWsp8xYJQRossRMWL0w9P8SiIc2IY/JrQrpz29nCwHMEEAEIAB0WIQS7I223sZ7vx= shH HWin83wZjDkxgQUCWzoywAAKCRCn83wZjDkxgQaDCACyFuBLQWNvLT8GTDqTf/gETzmtoEM6Y= r8O 4jbYg05xiFzAqMZctQsm3zHakx2JrimxDvQJRQJQzp5ICJ7J/BOuSL4FE1SPeQIfjm4jyBZGH= P/W vgHsT5e3+ZCPePPZO+3irarTKVhaaP70Tpka6EsOCZzO6L8D6tUDkhxMX0ymy7p8w9Yt1eD0o= Ume mxrKdS1ulpNJUTBw7gJN8bMowVnycEm6wntxOjrCxuwbkKhFLdn0ejcXQ0UkfbUFKfU64gGBu= S53 ZlM8XlOhQEIw/FrdXszhR+Tg3Ag130cmJhOrghgOBLzvJfUd6OvDT5VIz0QGbAm8SWlAIIms1= 9Z8 kBsLwsCOBBMBCAA4AhsDBQsJCAcCBhUKCQgLAgQWAgMBAh4BAheAFiEEchf7rIzpz2NEoWjla= A3B HVMLeiMFAl+04McACgkQaA3BHVMLeiPHXAf/SEoZa6CKoOAs1ctEP/hN8cEQqbEiuZ+71nm3u= /BQ p/CEUvqGq+iVc8kkpClDbPz5fa9mb+yWwufsnXKOs6ygmEoAEOL7dBZZIaRobBEkB09VXIkx8= lE0 00grBVtToHUGRfZcMoMZ98XhPGU6lJDN200j/2CV46hQDz6PLySecNjOME05mosbYW5N2JwFd= uXP Qx++DjWB32QLBhcOcP3WslTy3PKVe/TcTvk0JpPFMz4UFc+awBVhDgZiGGAW3xLZRYyhpoAEs= N7u XkV2ct0MRxuZ3y4tTYJobhbZwutRojiPPZduRw9CSpNDcQHruFiSOIQTpnLeCA6K2JAZyqmP/= 8LA lAQTAQgAPhYhBHIX+6yM6c9jRKFo5WgNwR1TC3ojBQJbOdLgAhsDBQkDwmcABQsJCAcCBhUKC= QgL AgQWAgMBAh4BAheAAAoJEGgNwR1TC3ojR80H/jH+vYavwQ+TvO8ksXL9JQWc3IFSiGpuSVXLC= dg6 2AmRirxW+qCwNncNQyb9rd30gzdectSkPWL3KSqEResBe24IbA5/jSkPweJasgXtfhuyoeCJ6= PXo clQQGKIoFIAEv1s8l0ggPZswvCinegl1diyJXUXmdEJRTWYAtxn/atut1o6Giv6D2qmYbXN7m= neM C5MzlLaJKUtoH7U/IjVw1sx2qtxAZGKVm4RZxPnMCp9E1MAr5t4dP5gJCIiqsdrVqI6iKupZs= tMx stPU//azmz7ZWWxT0JzgJqZSvPYx/SATeexTYBP47YFyri4jnsty2ErS91E6H8osBv6pnSn7e= ArO wE0EWznS4AEIAMYmP4M/V+T5RY5at/g7rUdNsLhWv1APYrh9RQefODYHrNRHUE9eosYbT6XMr= yR9 hT8XlGOYRwKWwiQBoWSDiTMo/Xi29jUnn4BXfI2px2DTXwc22LKtLAgTRjP+qbU63Y0xnQN29= UGD bYgyyK51DW3H0If2a3JNsheAAK+Xc9baj0LGIc8T9uiEWHBnCH+RdhgATnWWGKdDegUR5BkDf= Dg5 O/FISymJBHx2Dyoklv5g4BzkgqTqwmaYzsl8UxZKvbaxq0zbehDda8lvhFXodNFMAgTLJlLuD= YOG LK2AwbrS3Sp0AEbkpdJBb44qVlGm5bApZouHeJ/+n+7r12+lqdsAEQEAAcLAfAQYAQgAJhYhB= HIX +6yM6c9jRKFo5WgNwR1TC3ojBQJbOdLgAhsMBQkDwmcAAAoJEGgNwR1TC3ojpfcIAInwP5Olc= EKo kTnHCiDTz4Ony4GnHRP2fXATQZCKxmu4AJY2h9ifw9Nf2TjCZ6AMvC3thAN0rFDj55N9l4s1C= paD o4J+0fkrHuyNacnT206CeJV1E7NYntxUn+LSiRrOdywn6erjxRi9EYTVLCHcDhBEjKmFZfg4A= M4G ZMWX1lg0+eHbd5oL1as28WvvI/uIaMyV8RbyXot1r/8QLlWldU3NrTF5p7TMU2y3ZH2mf5suS= KHA MtbE4jKJ8ZHFOo3GhLgjVrBWHE9JXO08xKkgD+w6v83+nomsEuf6C6LYrqY/tsZvyEX6zN8Ct= irP dPWu/VXNRYAl/lat7lSI3H26qrHCwHYEGAEIACACGwwWIQRyF/usjOnPY0ShaOVoDcEdUwt6I= wUC X7Th+gAKCRBoDcEdUwt6I1UpCAC6QD5CCT0KFET29VVo4d+JdB5YJZmB72bMyVk12MGrx0fPp= OgW muh+7n1zULU4UciMSa0C1uq1eZZu044X3xIcVQp9+/nINM2d4+8yLEktK8f2jMIulYRSQOYX5= 909 E7a4oyj+AEgEDvcQ9WZPf+tSVpPDsrLiIezF5JdH8W0SlhPd13kAYSCR0H53PmAS6a3Z8UeOO= kof Fp/9g2GvHGr7ln9s6wenGbq7NOPMXtFG0luQDkqOFfTNI82COJ2ak5j8WMl25t9Cr5s9K2FUW= orW CeJMmGDIblC76yQzI2Ub2jXudrZCgLzHr3Bnbd/B+CklVwMpaskDluc7z3OP7KvXzsFNBF+04= RMB EADSYkyG/50PjD4CAbEnwQYZl61RVMBb3jug1j2dDPmOcQSYbw7pKLLLUoHCwnZXeEhRdiZ5w= qMo fW6DNo9irHecDc0815aN5RMxUQIqX7W6L4lI2qdKsaEcMtYQVdxbdf1cjcMGGO6oxlQE0FO3C= E/8 y4G/kndDT0APK9jb6SatTJyPJxXwNmR+HYrkbWr4RAVM6et0pQmMZ8sak+QEM0kBlKbI0sodQ= OXs 7Y9acMYJ5F+xfWU9JVAaOs2TOmQavDZMr6MeOpe/UKFjAZvcvv5yF8v7l3U+kWrpj4wc1bbRp= pDp 5k5RlCckAJ3QBNW6Im/YlgieJCK9ns0So1yVxrmjNZu4Un3UGFnndI+dvp1vahkjTZNRo2FzP= GUl k2oF+juZn26c0YAQSv16BPTVkTQGuFfrVF+P76qg5dJvjuBdhmLOhLu1i63nUYMIw7jBf1Qir= 0H6 98oyLfdGYtdHB0MN5IXSVpkgymIHp6b7H36OMeem4onzoeiMUv8IX+fwQMeJ8fBsl+yTtA1J1= Lon gjXxBkbvEG2AUjpDv73ZuMcu0WcHc5ajfM4wixTght+OgZ7Md5LtpTbV+qmaGo5nw+KDujdF3= bNq RMI7nwnyVvbAi3QZqiw+Tr7EZY8r+77ecNvF0hdbcoUGsSLPvLBSbcHwpYnbn36uehXBaxFws= bC0 KwARAQABwsKsBBgBCAAgFiEEchf7rIzpz2NEoWjlaA3BHVMLeiMFAl+04RMCGwICQAkQaA3BH= VML eiPBdCAEGQEIAB0WIQTGd2b9+m4xR3BgUWiWH8TcRCi/4AUCX7ThEwAKCRCWH8TcRCi/4JMoE= AC0 B6V5uOWXzAg8e7v3e60R/0exHxFeTgKEvfNSJM204K2/c+mGH4UKh/kdiJoKF/y6Lu80o8ndt= vyG mQJ0yhypFyBSDNbgdeOHagIt7VvwXovwknYg+w2NavdhG5envhqX902XQE0mNNH1XizDRZ422= 8wl V/fixCmAETZu5GKIFSTHOQrlZlwZ/RRRd3IqEb6bmpbPJZbBXOFolASXw3zs41xMar9zjLSY9= LKp SrF3Zy5F27QJhLxWsalLjtWcEwkxn5a5/4OAvAoNIn5NMGRoBmEYp3ZCRNbzkGT2gyOxrTcSf= g7M eHy//LX+1rpN02erf6HyoY6vLFFQlYZ5cnf97o67g7vz4cMGAbB4zZ/j5h3ggkSbjlsCmqXX4= x/Z 7bB1078lwQdg9x+zo+0vxzk10TU1FoWsSAiravlw0riSjr+wjXr5IzBeMm0A7K1MZ2lne+mKj= T72 bz2KE6isyrpIn4xlzLMQuh1OCthrHKb6GiJO4JWg/jWIt3yBov4EdFDXOnJl7Tq2CkpwrDTsH= Dgi Sw4xK+mHW3QFztsMQi6p7qaPCUSYAPIY4xzOTqExhMj/3Pgyg6hTvdGIpV9VzfMXIR68Pmw79= BGZ tEWaVXS+1ja+6kBOFIQ37sqOBorOGfp6c6Di+IN8mkEHyt9s2Q/YqpPdedm+58tinuaBSBeUk= mhM B/9rSzQXr84M1rhAehbaqDE+UYw8sfsQXiJkZYZz9BHp0YfhD9J5iwgGhYj+LU6dW1+eEsgcq= fMH Dexdvr9koHXeuMNaXa40++GX9cY9DU8RHbeYxGj35eVY793F4gfAAsD5DCF5pCcd7mhicp5eD= rIq 258874hM7w1aSwMS+0K+oppSlIfdogKM/l8RA6oDGZySTkwxXQ8RAYb/ddVc5Ua2gzXccODBz= JuT a+aY+BxmIdz4oMzaNCiK3gedGKoRjnrGguEO+YL/Ji3NhHOXRKu/QVTFDmgpmA0oQTfY+iaLz= bFk keWD97V4vo2flZ8ii37fcKNX4daGGnRRFUh4BrBG =3DskgE -----END PGP PUBLIC KEY BLOCK----- --------------23E1404708D2B7816DE29839-- --TtE0nB4yElvNRRDtk7uZnjah9k8KfGcy8-- --IXOfhhr90WUTexsH77RH73TvXPY0rJsGv Content-Type: application/pgp-signature; name="OpenPGP_signature.asc" Content-Description: OpenPGP digital signature Content-Disposition: attachment; filename="OpenPGP_signature" -----BEGIN PGP SIGNATURE----- wsF5BAABCAAjFiEExndm/fpuMUdwYFFolh/E3EQov+AFAl+/qC0FAwAAAAAACgkQlh/E3EQov+B9 4RAAmDhVfFHSJ2XTTw3bC+3YNacgAaiSHEAeZ7vWRysKqBASNMFu0NpBnTEbFoGMZ0JkUrvOVcKg AY/qYGZgFLIC/H2SeT5PpGru9k4qQ9sqYAAuQOrrWrwcP2NbSBnkGaHjWWoKrS3xTAFPSqkr90RP CVbrEvMVrjUEPOplh+tUlqrKN20Lvsc3ilaKxCE5oDqvYkyXGrxxpk56xi9g8IAK7aVAnnVQcEJV abantmzhBrpVCluIZJCoqPczk0dhSzERrHGoHvRJxMxnIwi2eEzHFnNcJfcXF4bmeIkHHrW0Don2 6wVRpKzgXkEnQ/fw1nZaDosWovwdzupA+QTEu8ah9fbB2F6LyapMjTEd2EbzlcAghFUMnnOFZqQw sI57NHz+YAQAo4bdIjlV5BFPs41Klas6VbvYyRW1eB+wnfieK678x1W/mGe5boErEQV5sH5KikLb 8FtgNWvkGOdm3zHID7jX24Wifn0OnfSamC/hbvvxKnBYHC5C6mwNpwzlxBleX86SK0wossNyceb3 T5BOorAbDGZ1QEfMnNYEhIyF8DLgQ//rYg4f3uMQmFpwECXpUPJyBqIZ3DIfB4brDu7kwuwhF4LM 2Snmqx06Pq8Azk9Bh3ZlFugUgqG/nttPVZOP55rB8sRisey9ns8XBOEKPMzziKCFHZxmzoVKAssp zBE= =qkyn -----END PGP SIGNATURE----- --IXOfhhr90WUTexsH77RH73TvXPY0rJsGv--