All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.sun.glass.ui.monocle.EPDSystem Maven / Gradle / Ivy

Go to download

Headless graphics driver for JavaFX. Cloned from https://github.com/sebivenlo/testfx-monocle and adapted for maven, java 17 and javafx 21

The newest version!
/*
 * Copyright (c) 2019, 2021, Oracle and/or its affiliates. All rights reserved.
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
 *
 * This code is free software; you can redistribute it and/or modify it
 * under the terms of the GNU General Public License version 2 only, as
 * published by the Free Software Foundation.  Oracle designates this
 * particular file as subject to the "Classpath" exception as provided
 * by Oracle in the LICENSE file that accompanied this code.
 *
 * This code is distributed in the hope that it will be useful, but WITHOUT
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
 * version 2 for more details (a copy is included in the LICENSE file that
 * accompanied this code).
 *
 * You should have received a copy of the GNU General Public License version
 * 2 along with this work; if not, write to the Free Software Foundation,
 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
 *
 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
 * or visit www.oracle.com if you need additional information or have any
 * questions.
 */
package com.sun.glass.ui.monocle;

import com.sun.glass.utils.NativeLibLoader;
import java.nio.ByteOrder;
import java.nio.IntBuffer;
import java.security.Permission;
import java.text.MessageFormat;

/**
 * A Java-language interface to the device API of the Electrophoretic Display
 * Controller (EPDC) frame buffer driver. {@code EPDSystem} is a singleton. Its
 * instance is obtained by calling the {@link EPDSystem#getEPDSystem} method.
 * This class also extends {@link LinuxSystem.FbVarScreenInfo} to provide all of
 * the fields in {@code fb_var_screeninfo}, defined in linux/fb.h.
 */
class EPDSystem {

    /**
     * The value for {@link FbVarScreenInfo#setActivate} to ensure that the EPDC
     * driver receives the initialization request.
     */
    static final int FB_ACTIVATE_FORCE = 128;

    /**
     * The value for {@link FbVarScreenInfo#setRotate} to set the frame buffer
     * rotation to un-rotated (upright landscape mode).
     */
    static final int FB_ROTATE_UR = 0;

    /**
     * The value for {@link FbVarScreenInfo#setRotate} to set the frame buffer
     * rotation to 90-degrees clockwise (upside-down portrait mode).
     */
    static final int FB_ROTATE_CW = 1;

    /**
     * The value for {@link FbVarScreenInfo#setRotate} to set the frame buffer
     * rotation to 180-degrees upside-down (upside-down landscape mode).
     */
    static final int FB_ROTATE_UD = 2;

    /**
     * The value for {@link FbVarScreenInfo#setRotate} to set the frame buffer
     * rotation to 90-degrees counter-clockwise (upright portrait mode).
     */
    static final int FB_ROTATE_CCW = 3;

    /**
     * The value for {@link FbVarScreenInfo#setGrayscale} to set the frame
     * buffer to an 8-bit grayscale pixel format.
     */
    static final int GRAYSCALE_8BIT = 0x1;

    /**
     * The value for {@link FbVarScreenInfo#setGrayscale} to set the frame
     * buffer to an inverted 8-bit grayscale pixel format.
     */
    static final int GRAYSCALE_8BIT_INVERTED = 0x2;

    /**
     * Region update mode, in which updates to the display must be submitted
     * with an IOCTL call to {@link #MXCFB_SEND_UPDATE}.
     */
    static final int AUTO_UPDATE_MODE_REGION_MODE = 0;

    /**
     * Automatic mode, in which updates are generated automatically by the
     * driver when it detects that pages in a frame buffer memory region have
     * been modified.
     */
    static final int AUTO_UPDATE_MODE_AUTOMATIC_MODE = 1;

    /**
     * Snapshot update scheme, which processes the contents of the frame buffer
     * immediately and stores the update in a memory buffer internal to the
     * driver. When the IOCTL call to {@link #MXCFB_SEND_UPDATE} returns, the
     * frame buffer region is free and can be modified without affecting the
     * update.
     */
    static final int UPDATE_SCHEME_SNAPSHOT = 0;

    /**
     * Queue update scheme, which uses a work queue to handle the processing of
     * updates asynchronously. When updates are submitted with an IOCTL call to
     * {@link #MXCFB_SEND_UPDATE}, they are added to the queue and processed in
     * order as the EPDC hardware resources become available. The frame buffer
     * contents processed and displayed, therefore, may not reflect what was
     * present in the frame buffer when the update was submitted.
     */
    static final int UPDATE_SCHEME_QUEUE = 1;

    /**
     * Queue and Merge update scheme, which adds a merging step to the Queue
     * update scheme. Before an update is added to the work queue, it is
     * compared with other pending updates. If a pending update matches the mode
     * and flags of the current update and also overlaps the update region, it
     * will be merged with the current update. After all such merges, the final
     * merged update is submitted to the queue.
     */
    static final int UPDATE_SCHEME_QUEUE_AND_MERGE = 2;

    /**
     * Partial update mode, which applies the waveform to only the pixels that
     * change in a given region.
     */
    static final int UPDATE_MODE_PARTIAL = 0x0;

    /**
     * Full update mode, which applies the waveform to all pixels in a given
     * region.
     */
    static final int UPDATE_MODE_FULL = 0x1;

    /**
     * Auto waveform mode, which requests the driver to select the actual
     * waveform mode automatically based on the contents of the updated region.
     */
    static final int WAVEFORM_MODE_AUTO = 257;

    /**
     * The temperature value that requests the driver to use the ambient
     * temperature of the device.
     */
    static final int TEMP_USE_AMBIENT = 0x1000;

    /**
     * An update flag to enable inversion of all pixels in the updated region.
     */
    static final int EPDC_FLAG_ENABLE_INVERSION = 0x01;

    /**
     * An update flag to enable black-and-white posterization of all pixels in
     * the updated region.
     */
    static final int EPDC_FLAG_FORCE_MONOCHROME = 0x02;

    /**
     * An update flag to enable dithering of an 8-bit grayscale frame buffer to
     * 1-bit black and white, if supported by the driver or hardware.
     */
    static final int EPDC_FLAG_USE_DITHERING_Y1 = 0x2000;

    /**
     * An update flag to enable dithering of an 8-bit grayscale frame buffer to
     * 4-bit grayscale, if supported by the driver or hardware.
     */
    static final int EPDC_FLAG_USE_DITHERING_Y4 = 0x4000;

    /**
     * The power-down delay value to disable the powering down of the EPDC and
     * display power supplies.
     */
    static final int FB_POWERDOWN_DISABLE = -1;

    /**
     * Initialization waveform (0x0...0xF to 0xF in ~4000 ms). Clears the screen
     * to all white.
     * 

* "A first exemplary drive scheme provides waveforms that may be used to * change the display state of a pixel from any initial display state to a * new display state of white. The first drive scheme may be referred to as * an initialization or 'INIT' drive scheme." [United States Patent * 9,280,955]

*/ static final int WAVEFORM_MODE_INIT = 0; /** * Direct update waveform (0x0...0xF to 0x0 or 0xF in ~260 ms). Changes gray * pixels to black or white. *

* "A second exemplary drive scheme provides waveforms that may be used to * change the display state of a pixel from any initial display state to a * new display state of either white or black. The second drive scheme may * be referred to as a 'DU' drive scheme." [United States Patent * 9,280,955]

*/ static final int WAVEFORM_MODE_DU = 1; /** * Gray 4-level waveform (0x0...0xF to 0x0, 0x5, 0xA, or 0xF in ~500 ms). * Supports 2-bit grayscale images and text with lower quality. *

* "A third exemplary drive scheme provides waveforms that may be used to * change the display state of a pixel from any initial display state to a * new display state. The initial state may be any four-bit (16 gray states) * value. The new display state may be any two-bit (4 gray states) value. * The third drive scheme may be referred to as a 'GC4' drive scheme." * [United States Patent 9,280,955]

*/ static final int WAVEFORM_MODE_GC4 = 3; /** * Gray 16-level waveform (0x0...0xF to 0x0...0xF in ~760 ms). Supports * 4-bit grayscale images and text with high quality. *

* "A fourth exemplary drive scheme provides waveforms that may be used to * change the display state of a pixel from any initial display state to a * new display state. The initial state may be any four-bit (16 gray states) * value. The new display state may be any four-bit (16 gray states) value. * The fourth drive scheme may be referred to as a 'GC16' drive scheme." * [United States Patent 9,280,955]

*/ static final int WAVEFORM_MODE_GC16 = 2; /** * Animation waveform (0x0 or 0xF to 0x0 or 0xF in ~120 ms). Provides a fast * 1-bit black-and-white animation mode of up to eight frames per second. *

* "A fifth exemplary drive scheme provides waveforms that may be used to * change the display state of a pixel from an initial display state to a * new display state. The initial state must be white or black. The new * display state may be black or white. The fifth drive scheme may be * referred to as an 'A2' drive scheme. An advantage of A2 waveforms is that * they have generally short waveform periods, providing rapid display * updates. A disadvantage of A2 waveforms is that there use may result in * ghosting artifacts." [United States Patent 9,280,955]

*/ static final int WAVEFORM_MODE_A2 = 4; private static final Permission PERMISSION = new RuntimePermission("loadLibrary.*"); private static final EPDSystem INSTANCE = new EPDSystem(); /** * Checks for permission to load native libraries if running under a * security manager. */ private static void checkPermissions() { @SuppressWarnings("removal") SecurityManager security = System.getSecurityManager(); if (security != null) { security.checkPermission(PERMISSION); } } /** * Obtains the single instance of {@code EPDSystem}. Calling this method * requires the "loadLibrary.*" {@code RuntimePermission}. The * {@link #loadLibrary} method must be called on the EPDSystem instance * before any system calls can be made using it. * * @return the {@code EPDSystem} instance */ static EPDSystem getEPDSystem() { checkPermissions(); return INSTANCE; } /** * The IOCTL request code to define a mapping for common waveform modes. */ final int MXCFB_SET_WAVEFORM_MODES; /** * The IOCTL request code to set the temperature used by the EPDC driver in * subsequent panel updates. */ final int MXCFB_SET_TEMPERATURE; /** * The IOCTL request code to select between automatic and region update * mode. */ final int MXCFB_SET_AUTO_UPDATE_MODE; /** * The IOCTL request code to update a region of the frame buffer to the * display. */ final int MXCFB_SEND_UPDATE; /** * The IOCTL request code to block and wait for a previous update to * complete. */ final int MXCFB_WAIT_FOR_UPDATE_COMPLETE; /** * The IOCTL request code to set the delay between the completion of all * updates in the driver and when the driver should power down the EPDC and * display power supplies. */ final int MXCFB_SET_PWRDOWN_DELAY; /** * The IOCTL request code to get the current power-down delay value from the * driver. */ final int MXCFB_GET_PWRDOWN_DELAY; /** * The IOCTL request code to select a scheme for the flow of updates within * the driver. */ final int MXCFB_SET_UPDATE_SCHEME; private final LinuxSystem system; /** * Creates the single instance of {@code EPDSystem}. */ private EPDSystem() { system = LinuxSystem.getLinuxSystem(); MXCFB_SET_WAVEFORM_MODES = system.IOW('F', 0x2B, MxcfbWaveformModes.BYTES); MXCFB_SET_TEMPERATURE = system.IOW('F', 0x2C, Integer.BYTES); MXCFB_SET_AUTO_UPDATE_MODE = system.IOW('F', 0x2D, Integer.BYTES); MXCFB_SEND_UPDATE = system.IOW('F', 0x2E, MxcfbUpdateData.BYTES); MXCFB_WAIT_FOR_UPDATE_COMPLETE = system.IOW('F', 0x2F, Integer.BYTES); MXCFB_SET_PWRDOWN_DELAY = system.IOW('F', 0x30, Integer.BYTES); MXCFB_GET_PWRDOWN_DELAY = system.IOR('F', 0x31, IntStructure.BYTES); MXCFB_SET_UPDATE_SCHEME = system.IOW('F', 0x32, Integer.BYTES); } /** * Loads the native libraries required to make system calls using this * {@code EPDSystem} instance. This method must be called before any other * instance methods of {@code EPDSystem}. If this method is called multiple * times, it has no effect after the first call. */ void loadLibrary() { NativeLibLoader.loadLibrary("glass_monocle_epd"); } /** * Calls the {@code ioctl} system function, passing a write integer * parameter. This method is more convenient than passing the pointer to an * {@code IntStructure} with {@link LinuxSystem#ioctl} and can be used when * the request code is created by {@link LinuxSystem#IOW} for setting an * integer value. * * @param fd an open file descriptor * @param request a device-dependent request code * @param value the integer value * @return 0 if successful; otherwise -1 with {@code errno} set * appropriately */ native int ioctl(long fd, int request, int value); /** * A structure for passing the pointer to an integer in an IOCTL call. */ static class IntStructure extends C.Structure { private static final int VALUE = 0; private static final int NUM_INTS = 1; private static final int BYTES = NUM_INTS * Integer.BYTES; private final IntBuffer data; IntStructure() { b.order(ByteOrder.nativeOrder()); data = b.asIntBuffer(); } @Override int sizeof() { return BYTES; } int get(long p) { return data.get(VALUE); } void set(long p, int value) { data.put(VALUE, value); } } /** * Wraps the C structure {@code mxcfb_waveform_modes}, defined in * mxcfb.h. */ static class MxcfbWaveformModes extends C.Structure { private static final int MODE_INIT = 0; private static final int MODE_DU = 1; private static final int MODE_GC4 = 2; private static final int MODE_GC8 = 3; private static final int MODE_GC16 = 4; private static final int MODE_GC32 = 5; private static final int NUM_INTS = 6; private static final int BYTES = NUM_INTS * Integer.BYTES; private final IntBuffer data; MxcfbWaveformModes() { b.order(ByteOrder.nativeOrder()); data = b.asIntBuffer(); } @Override int sizeof() { return BYTES; } int getModeInit(long p) { return data.get(MODE_INIT); } int getModeDu(long p) { return data.get(MODE_DU); } int getModeGc4(long p) { return data.get(MODE_GC4); } int getModeGc8(long p) { return data.get(MODE_GC8); } int getModeGc16(long p) { return data.get(MODE_GC16); } int getModeGc32(long p) { return data.get(MODE_GC32); } void setModes(long p, int init, int du, int gc4, int gc8, int gc16, int gc32) { data.put(MODE_INIT, init); data.put(MODE_DU, du); data.put(MODE_GC4, gc4); data.put(MODE_GC8, gc8); data.put(MODE_GC16, gc16); data.put(MODE_GC32, gc32); } @Override public String toString() { return MessageFormat.format( "{0}[mode_init={1} mode_du={2} mode_gc4={3} mode_gc8={4} mode_gc16={5} mode_gc32={6}]", getClass().getName(), getModeInit(p), getModeDu(p), getModeGc4(p), getModeGc8(p), getModeGc16(p), getModeGc32(p)); } } /** * Wraps the C structure {@code mxcfb_update_data}, defined in * mxcfb.h. */ static class MxcfbUpdateData extends C.Structure { private static final int UPDATE_REGION_TOP = 0; private static final int UPDATE_REGION_LEFT = 1; private static final int UPDATE_REGION_WIDTH = 2; private static final int UPDATE_REGION_HEIGHT = 3; private static final int WAVEFORM_MODE = 4; private static final int UPDATE_MODE = 5; private static final int UPDATE_MARKER = 6; private static final int TEMP = 7; private static final int FLAGS = 8; private static final int ALT_BUFFER_DATA_VIRT_ADDR = 9; private static final int ALT_BUFFER_DATA_PHYS_ADDR = 10; private static final int ALT_BUFFER_DATA_WIDTH = 11; private static final int ALT_BUFFER_DATA_HEIGHT = 12; private static final int ALT_BUFFER_DATA_ALT_UPDATE_REGION_TOP = 13; private static final int ALT_BUFFER_DATA_ALT_UPDATE_REGION_LEFT = 14; private static final int ALT_BUFFER_DATA_ALT_UPDATE_REGION_WIDTH = 15; private static final int ALT_BUFFER_DATA_ALT_UPDATE_REGION_HEIGHT = 16; private static final int NUM_INTS = 17; private static final int BYTES = NUM_INTS * Integer.BYTES; private final IntBuffer data; MxcfbUpdateData() { b.order(ByteOrder.nativeOrder()); data = b.asIntBuffer(); } @Override int sizeof() { return BYTES; } int getUpdateRegionTop(long p) { return data.get(UPDATE_REGION_TOP); } int getUpdateRegionLeft(long p) { return data.get(UPDATE_REGION_LEFT); } int getUpdateRegionWidth(long p) { return data.get(UPDATE_REGION_WIDTH); } int getUpdateRegionHeight(long p) { return data.get(UPDATE_REGION_HEIGHT); } int getWaveformMode(long p) { return data.get(WAVEFORM_MODE); } int getUpdateMode(long p) { return data.get(UPDATE_MODE); } int getUpdateMarker(long p) { return data.get(UPDATE_MARKER); } int getTemp(long p) { return data.get(TEMP); } int getFlags(long p) { return data.get(FLAGS); } long getAltBufferDataVirtAddr(long p) { return data.get(ALT_BUFFER_DATA_VIRT_ADDR); } long getAltBufferDataPhysAddr(long p) { return data.get(ALT_BUFFER_DATA_PHYS_ADDR); } int getAltBufferDataWidth(long p) { return data.get(ALT_BUFFER_DATA_WIDTH); } int getAltBufferDataHeight(long p) { return data.get(ALT_BUFFER_DATA_HEIGHT); } int getAltBufferDataAltUpdateRegionTop(long p) { return data.get(ALT_BUFFER_DATA_ALT_UPDATE_REGION_TOP); } int getAltBufferDataAltUpdateRegionLeft(long p) { return data.get(ALT_BUFFER_DATA_ALT_UPDATE_REGION_LEFT); } int getAltBufferDataAltUpdateRegionWidth(long p) { return data.get(ALT_BUFFER_DATA_ALT_UPDATE_REGION_WIDTH); } int getAltBufferDataAltUpdateRegionHeight(long p) { return data.get(ALT_BUFFER_DATA_ALT_UPDATE_REGION_HEIGHT); } void setUpdateRegion(long p, int top, int left, int width, int height) { data.put(UPDATE_REGION_TOP, top); data.put(UPDATE_REGION_LEFT, left); data.put(UPDATE_REGION_WIDTH, width); data.put(UPDATE_REGION_HEIGHT, height); } void setWaveformMode(long p, int mode) { data.put(WAVEFORM_MODE, mode); } void setUpdateMode(long p, int mode) { data.put(UPDATE_MODE, mode); } void setUpdateMarker(long p, int marker) { data.put(UPDATE_MARKER, marker); } void setTemp(long p, int temp) { data.put(TEMP, temp); } void setFlags(long p, int flags) { data.put(FLAGS, flags); } void setAltBufferData(long p, long virtAddr, long physAddr, int width, int height, int altUpdateRegionTop, int altUpdateRegionLeft, int altUpdateRegionWidth, int altUpdateRegionHeight) { data.put(ALT_BUFFER_DATA_VIRT_ADDR, (int) virtAddr); data.put(ALT_BUFFER_DATA_PHYS_ADDR, (int) physAddr); data.put(ALT_BUFFER_DATA_WIDTH, width); data.put(ALT_BUFFER_DATA_HEIGHT, height); data.put(ALT_BUFFER_DATA_ALT_UPDATE_REGION_TOP, altUpdateRegionTop); data.put(ALT_BUFFER_DATA_ALT_UPDATE_REGION_LEFT, altUpdateRegionLeft); data.put(ALT_BUFFER_DATA_ALT_UPDATE_REGION_WIDTH, altUpdateRegionWidth); data.put(ALT_BUFFER_DATA_ALT_UPDATE_REGION_HEIGHT, altUpdateRegionHeight); } @Override public String toString() { return MessageFormat.format( "{0}[update_region.top={1} update_region.left={2} update_region.width={3} update_region.height={4}" + " waveform_mode={5} update_mode={6} update_marker={7} temp={8} flags=0x{9}" + " alt_buffer_data.virt_addr=0x{10} alt_buffer_data.phys_addr=0x{11}" + " alt_buffer_data.width={12} alt_buffer_data.height={13}" + " alt_buffer_data.alt_update_region.top={14} alt_buffer_data.alt_update_region.left={15}" + " alt_buffer_data.alt_update_region.width={16} alt_buffer_data.alt_update_region.height={17}]", getClass().getName(), Integer.toUnsignedLong(getUpdateRegionTop(p)), Integer.toUnsignedLong(getUpdateRegionLeft(p)), Integer.toUnsignedLong(getUpdateRegionWidth(p)), Integer.toUnsignedLong(getUpdateRegionHeight(p)), Integer.toUnsignedLong(getWaveformMode(p)), Integer.toUnsignedLong(getUpdateMode(p)), Integer.toUnsignedLong(getUpdateMarker(p)), getTemp(p), Integer.toHexString(getFlags(p)), Long.toHexString(getAltBufferDataVirtAddr(p)), Long.toHexString(getAltBufferDataPhysAddr(p)), Integer.toUnsignedLong(getAltBufferDataWidth(p)), Integer.toUnsignedLong(getAltBufferDataHeight(p)), Integer.toUnsignedLong(getAltBufferDataAltUpdateRegionTop(p)), Integer.toUnsignedLong(getAltBufferDataAltUpdateRegionLeft(p)), Integer.toUnsignedLong(getAltBufferDataAltUpdateRegionWidth(p)), Integer.toUnsignedLong(getAltBufferDataAltUpdateRegionHeight(p))); } } /** * Wraps the entire C structure {@code fb_var_screeninfo}, defined in * linux/fb.h. */ static class FbVarScreenInfo extends LinuxSystem.FbVarScreenInfo { native int getGrayscale(long p); native int getRedOffset(long p); native int getRedLength(long p); native int getRedMsbRight(long p); native int getGreenOffset(long p); native int getGreenLength(long p); native int getGreenMsbRight(long p); native int getBlueOffset(long p); native int getBlueLength(long p); native int getBlueMsbRight(long p); native int getTranspOffset(long p); native int getTranspLength(long p); native int getTranspMsbRight(long p); native int getNonstd(long p); native int getActivate(long p); native int getHeight(long p); native int getWidth(long p); native int getAccelFlags(long p); native int getPixclock(long p); native int getLeftMargin(long p); native int getRightMargin(long p); native int getUpperMargin(long p); native int getLowerMargin(long p); native int getHsyncLen(long p); native int getVsyncLen(long p); native int getSync(long p); native int getVmode(long p); native int getRotate(long p); native void setGrayscale(long p, int grayscale); native void setNonstd(long p, int nonstd); native void setHeight(long p, int height); native void setWidth(long p, int width); native void setAccelFlags(long p, int accelFlags); native void setPixclock(long p, int pixclock); native void setLeftMargin(long p, int leftMargin); native void setRightMargin(long p, int rightMargin); native void setUpperMargin(long p, int upperMargin); native void setLowerMargin(long p, int lowerMargin); native void setHsyncLen(long p, int hsyncLen); native void setVsyncLen(long p, int vsyncLen); native void setSync(long p, int sync); native void setVmode(long p, int vmode); native void setRotate(long p, int rotate); } @Override public String toString() { return MessageFormat.format("{0}[MXCFB_SET_WAVEFORM_MODES=0x{1} MXCFB_SET_TEMPERATURE=0x{2} " + "MXCFB_SET_AUTO_UPDATE_MODE=0x{3} MXCFB_SEND_UPDATE=0x{4} MXCFB_WAIT_FOR_UPDATE_COMPLETE=0x{5} " + "MXCFB_SET_PWRDOWN_DELAY=0x{6} MXCFB_GET_PWRDOWN_DELAY=0x{7} MXCFB_SET_UPDATE_SCHEME=0x{8}]", getClass().getName(), Integer.toHexString(MXCFB_SET_WAVEFORM_MODES), Integer.toHexString(MXCFB_SET_TEMPERATURE), Integer.toHexString(MXCFB_SET_AUTO_UPDATE_MODE), Integer.toHexString(MXCFB_SEND_UPDATE), Integer.toHexString(MXCFB_WAIT_FOR_UPDATE_COMPLETE), Integer.toHexString(MXCFB_SET_PWRDOWN_DELAY), Integer.toHexString(MXCFB_GET_PWRDOWN_DELAY), Integer.toHexString(MXCFB_SET_UPDATE_SCHEME) ); } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy