2018-07-21 01:44:44 +01:00
|
|
|
{ stdenv, fetchurl, lib
|
|
|
|
, pkgconfig, intltool, autoreconfHook
|
2017-10-07 14:19:13 +01:00
|
|
|
, file, expat, libdrm, xorg, wayland, wayland-protocols, openssl
|
2018-02-09 04:05:38 +00:00
|
|
|
, llvmPackages, libffi, libomxil-bellagio, libva-minimal
|
2017-10-07 14:19:13 +01:00
|
|
|
, libelf, libvdpau, valgrind-light, python2
|
2018-03-17 15:44:40 +00:00
|
|
|
, libglvnd
|
2017-10-07 14:19:13 +01:00
|
|
|
, enableRadv ? true
|
2017-02-15 13:25:16 +00:00
|
|
|
, galliumDrivers ? null
|
|
|
|
, driDrivers ? null
|
|
|
|
, vulkanDrivers ? null
|
2013-05-16 16:16:02 +01:00
|
|
|
}:
|
2004-06-09 18:53:30 +01:00
|
|
|
|
2013-05-16 16:16:02 +01:00
|
|
|
/** Packaging design:
|
2018-03-05 13:53:38 +00:00
|
|
|
- The basic mesa ($out) contains headers and libraries (GLU is in libGLU now).
|
2014-01-28 10:34:05 +00:00
|
|
|
This or the mesa attribute (which also contains GLU) are small (~ 2 MB, mostly headers)
|
2013-05-16 16:16:02 +01:00
|
|
|
and are designed to be the buildInput of other packages.
|
2015-02-22 18:11:38 +00:00
|
|
|
- DRI drivers are compiled into $drivers output, which is much bigger and
|
|
|
|
depends on LLVM. These should be searched at runtime in
|
|
|
|
"/run/opengl-driver{,-32}/lib/*" and so are kind-of impure (given by NixOS).
|
2013-05-16 16:16:02 +01:00
|
|
|
(I suppose on non-NixOS one would create the appropriate symlinks from there.)
|
2014-01-28 10:34:05 +00:00
|
|
|
- libOSMesa is in $osmesa (~4 MB)
|
2013-05-16 16:16:02 +01:00
|
|
|
*/
|
|
|
|
|
2016-08-24 12:35:30 +01:00
|
|
|
with stdenv.lib;
|
|
|
|
|
2018-08-20 20:11:29 +01:00
|
|
|
if ! lists.elem stdenv.hostPlatform.system platforms.mesaPlatforms then
|
2016-08-24 12:35:30 +01:00
|
|
|
throw "unsupported platform for Mesa"
|
|
|
|
else
|
2015-04-14 21:41:22 +01:00
|
|
|
|
2017-02-15 13:25:16 +00:00
|
|
|
let
|
|
|
|
defaultGalliumDrivers =
|
treewide: isArm -> isAarch32
Following legacy packing conventions, `isArm` was defined just for
32-bit ARM instruction set. This is confusing to non packagers though,
because Aarch64 is an ARM instruction set.
The official ARM overview for ARMv8[1] is surprisingly not confusing,
given the overall state of affairs for ARM naming conventions, and
offers us a solution. It divides the nomenclature into three levels:
```
ISA: ARMv8 {-A, -R, -M}
/ \
Mode: Aarch32 Aarch64
| / \
Encoding: A64 A32 T32
```
At the top is the overall v8 instruction set archicture. Second are the
two modes, defined by bitwidth but differing in other semantics too, and
buttom are the encodings, (hopefully?) isomorphic if they encode the
same mode.
The 32 bit encodings are mostly backwards compatible with previous
non-Thumb and Thumb encodings, and if so we can pun the mode names to
instead mean "sets of compatable or isomorphic encodings", and then
voilà we have nice names for 32-bit and 64-bit arm instruction sets
which do not use the word ARM so as to not confused either laymen or
experienced ARM packages.
[1]: https://developer.arm.com/products/architecture/a-profile
2018-03-20 02:41:06 +00:00
|
|
|
if stdenv.isAarch32
|
2018-05-26 16:03:44 +01:00
|
|
|
then ["virgl" "nouveau" "freedreno" "vc4" "etnaviv" "imx"]
|
2017-03-29 11:02:36 +01:00
|
|
|
else if stdenv.isAarch64
|
2018-05-26 16:03:44 +01:00
|
|
|
then ["virgl" "nouveau" "vc4" ]
|
|
|
|
else ["virgl" "svga" "i915" "r300" "r600" "radeonsi" "nouveau"];
|
2017-02-15 13:25:16 +00:00
|
|
|
defaultDriDrivers =
|
treewide: isArm -> isAarch32
Following legacy packing conventions, `isArm` was defined just for
32-bit ARM instruction set. This is confusing to non packagers though,
because Aarch64 is an ARM instruction set.
The official ARM overview for ARMv8[1] is surprisingly not confusing,
given the overall state of affairs for ARM naming conventions, and
offers us a solution. It divides the nomenclature into three levels:
```
ISA: ARMv8 {-A, -R, -M}
/ \
Mode: Aarch32 Aarch64
| / \
Encoding: A64 A32 T32
```
At the top is the overall v8 instruction set archicture. Second are the
two modes, defined by bitwidth but differing in other semantics too, and
buttom are the encodings, (hopefully?) isomorphic if they encode the
same mode.
The 32 bit encodings are mostly backwards compatible with previous
non-Thumb and Thumb encodings, and if so we can pun the mode names to
instead mean "sets of compatable or isomorphic encodings", and then
voilà we have nice names for 32-bit and 64-bit arm instruction sets
which do not use the word ARM so as to not confused either laymen or
experienced ARM packages.
[1]: https://developer.arm.com/products/architecture/a-profile
2018-03-20 02:41:06 +00:00
|
|
|
if (stdenv.isAarch32 || stdenv.isAarch64)
|
2017-02-15 13:25:16 +00:00
|
|
|
then ["nouveau"]
|
|
|
|
else ["i915" "i965" "nouveau" "radeon" "r200"];
|
|
|
|
defaultVulkanDrivers =
|
treewide: isArm -> isAarch32
Following legacy packing conventions, `isArm` was defined just for
32-bit ARM instruction set. This is confusing to non packagers though,
because Aarch64 is an ARM instruction set.
The official ARM overview for ARMv8[1] is surprisingly not confusing,
given the overall state of affairs for ARM naming conventions, and
offers us a solution. It divides the nomenclature into three levels:
```
ISA: ARMv8 {-A, -R, -M}
/ \
Mode: Aarch32 Aarch64
| / \
Encoding: A64 A32 T32
```
At the top is the overall v8 instruction set archicture. Second are the
two modes, defined by bitwidth but differing in other semantics too, and
buttom are the encodings, (hopefully?) isomorphic if they encode the
same mode.
The 32 bit encodings are mostly backwards compatible with previous
non-Thumb and Thumb encodings, and if so we can pun the mode names to
instead mean "sets of compatable or isomorphic encodings", and then
voilà we have nice names for 32-bit and 64-bit arm instruction sets
which do not use the word ARM so as to not confused either laymen or
experienced ARM packages.
[1]: https://developer.arm.com/products/architecture/a-profile
2018-03-20 02:41:06 +00:00
|
|
|
if (stdenv.isAarch32 || stdenv.isAarch64)
|
2017-02-15 13:25:16 +00:00
|
|
|
then []
|
|
|
|
else ["intel"] ++ lib.optional enableRadv "radeon";
|
|
|
|
in
|
|
|
|
|
|
|
|
let gallium_ = galliumDrivers; dri_ = driDrivers; vulkan_ = vulkanDrivers; in
|
|
|
|
|
|
|
|
let
|
|
|
|
galliumDrivers =
|
2017-10-10 19:34:38 +01:00
|
|
|
(if gallium_ == null
|
2017-02-15 13:25:16 +00:00
|
|
|
then defaultGalliumDrivers
|
|
|
|
else gallium_)
|
2018-03-17 15:45:44 +00:00
|
|
|
++ ["swrast" "virgl"];
|
2017-02-15 13:25:16 +00:00
|
|
|
driDrivers =
|
|
|
|
(if dri_ == null
|
|
|
|
then defaultDriDrivers
|
|
|
|
else dri_) ++ ["swrast"];
|
|
|
|
vulkanDrivers =
|
|
|
|
if vulkan_ == null
|
|
|
|
then defaultVulkanDrivers
|
|
|
|
else vulkan_;
|
|
|
|
in
|
|
|
|
|
2013-01-28 16:38:42 +00:00
|
|
|
let
|
2018-08-26 09:49:12 +01:00
|
|
|
version = "18.1.7";
|
2016-08-24 12:35:30 +01:00
|
|
|
branch = head (splitString "." version);
|
2013-01-28 16:38:42 +00:00
|
|
|
in
|
2013-10-05 20:18:07 +01:00
|
|
|
|
2018-03-17 15:47:55 +00:00
|
|
|
let self = stdenv.mkDerivation {
|
2013-03-24 11:36:33 +00:00
|
|
|
name = "mesa-noglu-${version}";
|
2010-04-29 15:57:02 +01:00
|
|
|
|
2013-10-05 20:18:07 +01:00
|
|
|
src = fetchurl {
|
2015-01-24 21:37:16 +00:00
|
|
|
urls = [
|
2017-03-07 20:48:18 +00:00
|
|
|
"ftp://ftp.freedesktop.org/pub/mesa/mesa-${version}.tar.xz"
|
2015-03-28 20:57:03 +00:00
|
|
|
"ftp://ftp.freedesktop.org/pub/mesa/${version}/mesa-${version}.tar.xz"
|
2016-08-24 12:35:30 +01:00
|
|
|
"ftp://ftp.freedesktop.org/pub/mesa/older-versions/${branch}.x/${version}/mesa-${version}.tar.xz"
|
2017-11-13 17:49:46 +00:00
|
|
|
"https://mesa.freedesktop.org/archive/mesa-${version}.tar.xz"
|
2015-01-24 21:37:16 +00:00
|
|
|
];
|
2018-08-26 09:49:12 +01:00
|
|
|
sha256 = "655e3b32ce3bdddd5e6e8768596e5d4bdef82d0dd37067c324cc4b2daa207306";
|
2007-02-26 17:05:27 +00:00
|
|
|
};
|
2010-01-13 12:43:17 +00:00
|
|
|
|
2012-01-16 17:48:33 +00:00
|
|
|
prePatch = "patchShebangs .";
|
2011-07-14 13:22:55 +01:00
|
|
|
|
2016-08-24 12:35:30 +01:00
|
|
|
# TODO:
|
|
|
|
# revive ./dricore-gallium.patch when it gets ported (from Ubuntu), as it saved
|
|
|
|
# ~35 MB in $drivers; watch https://launchpad.net/ubuntu/+source/mesa/+changelog
|
2014-01-28 10:34:05 +00:00
|
|
|
patches = [
|
2016-05-22 17:05:12 +01:00
|
|
|
./symlink-drivers.patch
|
2018-07-18 15:41:21 +01:00
|
|
|
./missing-includes.patch # dev_t needs sys/stat.h, time_t needs time.h, etc.-- fixes build w/musl
|
2018-08-06 20:23:02 +01:00
|
|
|
./disk_cache-include-dri-driver-path-in-cache-key.patch
|
2018-04-05 17:39:51 +01:00
|
|
|
];
|
2013-05-16 16:16:02 +01:00
|
|
|
|
2016-08-29 01:30:01 +01:00
|
|
|
outputs = [ "out" "dev" "drivers" "osmesa" ];
|
2013-05-16 16:16:02 +01:00
|
|
|
|
2016-08-24 12:35:30 +01:00
|
|
|
# TODO: Figure out how to enable opencl without having a runtime dependency on clang
|
2013-10-05 20:18:07 +01:00
|
|
|
configureFlags = [
|
2018-03-17 15:44:40 +00:00
|
|
|
"--sysconfdir=${libglvnd.driverLink}/etc"
|
2015-03-28 20:57:03 +00:00
|
|
|
"--localstatedir=/var"
|
2013-05-16 16:16:02 +01:00
|
|
|
"--with-dri-driverdir=$(drivers)/lib/dri"
|
2018-03-17 15:44:40 +00:00
|
|
|
"--with-dri-searchpath=${libglvnd.driverLink}/lib/dri"
|
2017-05-27 10:25:23 +01:00
|
|
|
"--with-platforms=x11,wayland,drm"
|
2018-08-25 17:34:41 +01:00
|
|
|
"--enable-texture-float"
|
2017-02-15 13:25:16 +00:00
|
|
|
]
|
|
|
|
++ (optional (galliumDrivers != [])
|
|
|
|
("--with-gallium-drivers=" +
|
|
|
|
builtins.concatStringsSep "," galliumDrivers))
|
|
|
|
++ (optional (driDrivers != [])
|
|
|
|
("--with-dri-drivers=" +
|
|
|
|
builtins.concatStringsSep "," driDrivers))
|
|
|
|
++ (optional (vulkanDrivers != [])
|
|
|
|
("--with-vulkan-drivers=" +
|
|
|
|
builtins.concatStringsSep "," vulkanDrivers))
|
|
|
|
++ [
|
2016-08-24 12:35:30 +01:00
|
|
|
(enableFeature stdenv.isLinux "dri3")
|
|
|
|
(enableFeature stdenv.isLinux "nine") # Direct3D in Wine
|
2018-03-17 15:44:40 +00:00
|
|
|
"--enable-libglvnd"
|
2016-08-24 12:35:30 +01:00
|
|
|
"--enable-dri"
|
|
|
|
"--enable-driglx-direct"
|
2015-03-28 20:57:03 +00:00
|
|
|
"--enable-gles1"
|
|
|
|
"--enable-gles2"
|
|
|
|
"--enable-glx"
|
2018-04-03 22:45:43 +01:00
|
|
|
# https://bugs.freedesktop.org/show_bug.cgi?id=35268
|
|
|
|
(enableFeature (!stdenv.hostPlatform.isMusl) "glx-tls")
|
2015-03-28 20:57:03 +00:00
|
|
|
"--enable-gallium-osmesa" # used by wine
|
2017-05-27 10:25:23 +01:00
|
|
|
"--enable-llvm"
|
2015-03-28 20:57:03 +00:00
|
|
|
"--enable-egl"
|
2013-05-22 16:09:00 +01:00
|
|
|
"--enable-xa" # used in vmware driver
|
2015-03-28 20:57:03 +00:00
|
|
|
"--enable-gbm"
|
|
|
|
"--enable-xvmc"
|
2013-11-16 20:23:10 +00:00
|
|
|
"--enable-vdpau"
|
2015-03-28 20:57:03 +00:00
|
|
|
"--enable-shared-glapi"
|
|
|
|
"--enable-llvm-shared-libs"
|
2018-01-21 10:01:47 +00:00
|
|
|
"--enable-omx-bellagio"
|
2016-08-24 12:35:30 +01:00
|
|
|
"--enable-va"
|
|
|
|
"--disable-opencl"
|
|
|
|
];
|
2012-01-16 17:48:33 +00:00
|
|
|
|
2017-09-19 18:20:24 +01:00
|
|
|
nativeBuildInputs = [ autoreconfHook intltool pkgconfig file ];
|
2013-05-16 16:16:02 +01:00
|
|
|
|
2016-08-24 12:35:30 +01:00
|
|
|
propagatedBuildInputs = with xorg;
|
|
|
|
[ libXdamage libXxf86vm ]
|
2015-04-14 21:41:22 +01:00
|
|
|
++ optional stdenv.isLinux libdrm;
|
2015-03-28 20:57:03 +00:00
|
|
|
|
2013-05-16 16:16:02 +01:00
|
|
|
buildInputs = with xorg; [
|
2018-03-17 15:44:40 +00:00
|
|
|
expat llvmPackages.llvm libglvnd
|
2014-04-19 12:03:50 +01:00
|
|
|
glproto dri2proto dri3proto presentproto
|
|
|
|
libX11 libXext libxcb libXt libXfixes libxshmfence
|
2017-10-07 14:19:13 +01:00
|
|
|
libffi wayland wayland-protocols libvdpau libelf libXvMC
|
2018-02-09 04:05:38 +00:00
|
|
|
libomxil-bellagio libva-minimal libpthreadstubs openssl/*or another sha1 provider*/
|
2018-06-26 10:54:03 +01:00
|
|
|
valgrind-light python2 python2.pkgs.Mako
|
2016-11-14 22:53:58 +00:00
|
|
|
];
|
2010-04-29 15:57:02 +01:00
|
|
|
|
2011-02-18 09:16:11 +00:00
|
|
|
enableParallelBuilding = true;
|
2015-02-22 18:11:38 +00:00
|
|
|
doCheck = false;
|
2013-05-16 16:16:02 +01:00
|
|
|
|
2015-04-15 05:38:47 +01:00
|
|
|
installFlags = [
|
2018-04-21 03:39:08 +01:00
|
|
|
"sysconfdir=\${drivers}/etc"
|
2015-04-15 05:38:47 +01:00
|
|
|
"localstatedir=\${TMPDIR}"
|
2018-03-17 15:44:40 +00:00
|
|
|
"vendorjsondir=\${out}/share/glvnd/egl_vendor.d"
|
2015-04-15 05:38:47 +01:00
|
|
|
];
|
2013-05-16 16:16:02 +01:00
|
|
|
|
2016-08-24 12:35:30 +01:00
|
|
|
# TODO: probably not all .la files are completely fixed, but it shouldn't matter;
|
|
|
|
postInstall = ''
|
|
|
|
# move gallium-related stuff to $drivers, so $out doesn't depend on LLVM
|
|
|
|
mv -t "$drivers/lib/" \
|
|
|
|
$out/lib/libXvMC* \
|
|
|
|
$out/lib/d3d \
|
|
|
|
$out/lib/vdpau \
|
|
|
|
$out/lib/bellagio \
|
|
|
|
$out/lib/libxatracker* \
|
2016-11-21 20:14:19 +00:00
|
|
|
$out/lib/libvulkan_*
|
2016-10-05 23:33:10 +01:00
|
|
|
|
2018-03-17 15:44:40 +00:00
|
|
|
# Move other drivers to a separate output
|
2016-11-21 20:14:19 +00:00
|
|
|
mv $out/lib/dri/* $drivers/lib/dri # */
|
2016-10-05 23:33:10 +01:00
|
|
|
rmdir "$out/lib/dri"
|
2018-03-17 15:44:40 +00:00
|
|
|
mv $out/lib/lib*_mesa* $drivers/lib
|
2013-05-22 16:09:00 +01:00
|
|
|
|
2016-08-24 12:35:30 +01:00
|
|
|
# move libOSMesa to $osmesa, as it's relatively big
|
2015-10-04 08:52:46 +01:00
|
|
|
mkdir -p {$osmesa,$drivers}/lib/
|
2016-08-24 12:35:30 +01:00
|
|
|
mv -t $osmesa/lib/ $out/lib/libOSMesa*
|
2013-11-17 09:53:41 +00:00
|
|
|
|
2016-08-24 12:35:30 +01:00
|
|
|
# now fix references in .la files
|
|
|
|
sed "/^libdir=/s,$out,$osmesa," -i $osmesa/lib/libOSMesa*.la
|
2013-11-17 09:53:41 +00:00
|
|
|
|
2016-08-24 12:35:30 +01:00
|
|
|
# set the default search path for DRI drivers; used e.g. by X server
|
2018-03-17 15:44:40 +00:00
|
|
|
substituteInPlace "$dev/lib/pkgconfig/dri.pc" --replace '$(drivers)' "${libglvnd.driverLink}"
|
|
|
|
|
|
|
|
# remove GLES libraries; they are provided by libglvnd
|
|
|
|
rm $out/lib/lib{GLESv1_CM,GLESv2}.*
|
|
|
|
|
|
|
|
# remove pkgconfig files for GL/GLES/EGL; they are provided by libGL.
|
|
|
|
rm $dev/lib/pkgconfig/{gl,egl,glesv1_cm,glesv2}.pc
|
|
|
|
|
|
|
|
# move vendor files
|
2017-01-27 11:51:16 +00:00
|
|
|
mv $out/share/ $drivers/
|
2018-03-17 15:44:40 +00:00
|
|
|
|
|
|
|
# Update search path used by glvnd
|
|
|
|
for js in $drivers/share/glvnd/egl_vendor.d/*.json; do
|
|
|
|
substituteInPlace "$js" --replace '"libEGL_' '"'"$drivers/lib/libEGL_"
|
|
|
|
done
|
2018-06-24 15:10:05 +01:00
|
|
|
|
|
|
|
# Update search path used by pkg-config
|
2018-07-16 16:47:24 +01:00
|
|
|
for pc in $dev/lib/pkgconfig/{d3d,dri,xatracker}.pc; do
|
2018-06-24 15:10:05 +01:00
|
|
|
substituteInPlace "$pc" --replace $out $drivers
|
|
|
|
done
|
2018-03-17 15:44:40 +00:00
|
|
|
'' + optionalString (vulkanDrivers != []) ''
|
2017-01-27 11:51:16 +00:00
|
|
|
# Update search path used by Vulkan (it's pointing to $out but
|
|
|
|
# drivers are in $drivers)
|
|
|
|
for js in $drivers/share/vulkan/icd.d/*.json; do
|
|
|
|
substituteInPlace "$js" --replace "$out" "$drivers"
|
|
|
|
done
|
2015-04-14 21:41:22 +01:00
|
|
|
'';
|
2013-05-22 16:09:00 +01:00
|
|
|
|
2016-08-24 12:35:30 +01:00
|
|
|
# TODO:
|
|
|
|
# check $out doesn't depend on llvm: builder failures are ignored
|
|
|
|
# for some reason grep -qv '${llvmPackages.llvm}' -R "$out";
|
|
|
|
postFixup = ''
|
2015-04-14 21:41:22 +01:00
|
|
|
# add RPATH so the drivers can find the moved libgallium and libdricore9
|
|
|
|
# moved here to avoid problems with stripping patchelfed files
|
2013-05-16 16:16:02 +01:00
|
|
|
for lib in $drivers/lib/*.so* $drivers/lib/*/*.so*; do
|
|
|
|
if [[ ! -L "$lib" ]]; then
|
|
|
|
patchelf --set-rpath "$(patchelf --print-rpath $lib):$drivers/lib" "$lib"
|
|
|
|
fi
|
|
|
|
done
|
|
|
|
'';
|
2011-02-18 09:16:11 +00:00
|
|
|
|
2018-03-17 15:44:40 +00:00
|
|
|
passthru = {
|
|
|
|
inherit libdrm version;
|
|
|
|
inherit (libglvnd) driverLink;
|
2018-03-17 15:47:55 +00:00
|
|
|
|
|
|
|
stubs = stdenv.mkDerivation {
|
|
|
|
name = "libGL-${libglvnd.version}";
|
|
|
|
outputs = [ "out" "dev" ];
|
|
|
|
|
|
|
|
# Use stub libraries from libglvnd and headers from Mesa.
|
|
|
|
buildCommand = ''
|
|
|
|
ln -s ${libglvnd.out} $out
|
|
|
|
mkdir -p $dev/{,lib/pkgconfig,nix-support}
|
|
|
|
echo "$out" > $dev/nix-support/propagated-build-inputs
|
|
|
|
ln -s ${self.dev}/include $dev/include
|
|
|
|
|
|
|
|
genPkgConfig() {
|
|
|
|
local name="$1"
|
|
|
|
local lib="$2"
|
|
|
|
|
|
|
|
cat <<EOF >$dev/lib/pkgconfig/$name.pc
|
|
|
|
Name: $name
|
|
|
|
Description: $lib library
|
|
|
|
Version: ${self.version}
|
|
|
|
Libs: -L${libglvnd.out}/lib -l$lib
|
|
|
|
Cflags: -I${self.dev}/include
|
|
|
|
EOF
|
|
|
|
}
|
|
|
|
|
|
|
|
genPkgConfig gl GL
|
|
|
|
genPkgConfig egl EGL
|
|
|
|
genPkgConfig glesv1_cm GLESv1_CM
|
|
|
|
genPkgConfig glesv2 GLESv2
|
|
|
|
'';
|
|
|
|
};
|
2018-03-17 15:44:40 +00:00
|
|
|
};
|
2010-04-29 15:57:02 +01:00
|
|
|
|
2015-04-14 21:41:22 +01:00
|
|
|
meta = with stdenv.lib; {
|
2008-03-01 17:44:50 +00:00
|
|
|
description = "An open source implementation of OpenGL";
|
2018-01-05 19:42:46 +00:00
|
|
|
homepage = https://www.mesa3d.org/;
|
2015-04-14 21:41:22 +01:00
|
|
|
license = licenses.mit; # X11 variant, in most files
|
2018-03-17 15:44:40 +00:00
|
|
|
platforms = platforms.linux;
|
2018-07-11 22:22:57 +01:00
|
|
|
maintainers = with maintainers; [ vcunat ];
|
2007-11-05 23:59:55 +00:00
|
|
|
};
|
2018-03-17 15:47:55 +00:00
|
|
|
};
|
|
|
|
in self
|