Commit 8fa612d1 authored by Reinhard Tartler's avatar Reinhard Tartler

Imported nxcompshad-3.1.0-2.tar.gz

Summary: Imported nxcompshad-3.1.0-2.tar.gz Keywords: Imported nxcompshad-3.1.0-2.tar.gz into Git repository
parent a840692e
ChangeLog:
nxcompshad-3.1.0-2
- Updated file VERSION to match the current release version.
nxcompshad-3.1.0-1
- Opened the 3.1.0 branch based on nxcompshad-3.0.0-19.
nxcompshad-3.0.0-19
- Skip initialization of damage extension if it was already disabled.
nxcompshad-3.0.0-18
- The poller destructor checks if the damage extension is in use.
- Small changes to the function getting the screen content in the case
damage is not in use.
nxcompshad-3.0.0-17
- Cleanup of some log messages.
nxcompshad-3.0.0-16
- Disabled some log message in the functions initializing the poller.
nxcompshad-3.0.0-15
- Before calling XTest functions, it is checked if the connection to
master X server has been initialized.
nxcompshad-3.0.0-14
- After the shm segment is created, its mode is changed and it is
marked for destroying. A check on the number of attaches is done.
nxcompshad-3.0.0-13
- Creating the shm segment even if the uid of master X server can't be
retrieved.
- Fixed reallocation of update region.
- Checking if the master X server provides XTest extension.
nxcompshad-3.0.0-12
- Fixed a compiler warning on AMD64 platform.
- Changed configure script to add -fPIC option.
- Fixed a mismatch in UpdateManager destructor.
nxcompshad-3.0.0-11
- Fixed the function setting the uid of shm segment.
- Sync with the master X server before marking the shm segment to be
destroyed.
nxcompshad-3.0.0-10
- Setting the shm segments as readable only by the master X server
process.
- Mark shm segments to be destroyed when the last process detach.
- Enabled keycode translation in order to allow keyboards of different
models to work.
nxcompshad-3.0.0-9
- Changed the LICENSE file to state that the software is only made
available under the version 2 of the GPL.
- Added file COPYING.
- Changes to translate keycodes between different keyboard types. The
keycodes are translated through the keysym.
- Changes to convert a Mode_switch key to ISO_Level3_Shift if it is
needed.
nxcompshad-3.0.0-8
- Added interface function NXShadowDisableShm disabling the use of
MIT-SHM extension.
- Added interface function NXShadowDisableDamage disabling the use of
DAMAGE extension.
- Added interface function NXShadowSetDisplayUid setting the uid of
shadowed X server
- Changed the owner of shared memory segments to the uid of the sha-
dowed X server.
- Fixed logWarning method.
- Moved the code initializing the use of shared memory to shmInit
method of X11 Poller.
nxcompshad-3.0.0-7
- Removed the class qualifier in the declaration of destroyUpdate-
ManagerRegion().
nxcompshad-3.0.0-6
- Fixed build errors on 64 bit platforms.
- Called XTestGrabControl to override possible grabs of the X server.
nxcompshad-3.0.0-5
- Added some stub members to the Poller class. These are intended to
handle keyboard state.
- Changes in the default polling algorithm to disable the line pri-
ority persistence.
nxcompshad-3.0.0-4
- If a low layer handling of screen changes is available (DAMAGE in
case of X11), polling routine uses it and returns immediately.
- Creating a Damage notify object if the extension is supported.
- DamageNotify events are saved in the update region. After all avail-
able events have been handled, the content of screen is retrieved
by ShnGetImage requests.
- XDamageSubtract and XSync are done before any event handling.
- Damages are requested as raw rectangles.
- Added Xdamage and Xrandr to required libraries.
- Fixed a problem with some lines not refreshed in shadowing mode.
nxcompshad-3.0.0-3
- Added destroyUpdateManagerRegion() method to UpdateManager class.
- Turned off some log messages.
- Changed configure.in to remove warnings related to deprecated header
files and options.
- Changed Makefile.in to remove autom4te.cache dir if the target is
'distclean'.
- Removed multi-word file names.
nxcompshad-3.0.0-2
- Changes to get the screen of original display by a ShmGetImage.
- Exit the polling loop if more than 50 ms have elapsed.
nxcompshad-3.0.0-1
- Created nxcompshad based on nxshadow-3.0.0-7.
nxshadow-3.0.0-7
- Deleted files KeyCursorTmp.cpp, scancodes.h, constant.h.
- Renamed NXshadow.h to Shadow.h.
- Merged NXshadowEvent.h with Shadow.h.
- Fixed configure.in, changed Makefile.in to build Xcompshad library,
rebuilt configure script.
nxshadow-3.0.0-6
- Implemented a callback mechanism to ask the caller program if screen
polling have to be suspended.
nxshadow-3.0.0-5
- Changes to comply with the library name.
- Fixed a bug in CorePoller that could prevent the update of the last
line of a rectangle.
nxshadow-3.0.0-4
- Removed some parameters of the NXShadowAddUpdaterDisplay() function.
nxshadow-3.0.0-3
- Updated copyright notices to the current year.
- Code cleanup in KeysCursorTmp.c file.
nxshadow-3.0.0-2
- If master agent uses shared memory extension, the GetImage is rea-
lized by XShmGetImage() function.
- Added new parameters to NXShadowAddUpdaterDisplay() function, the
depth and bit mask color of the master display.
nxshadow-3.0.0-1
- Opened the nxshadow-3.0.0 branch based on the nxaccess-3.0.0-13.
- Renamed NXaccess.h NXaccessEvent.h and RegionM.h files as NXshadow.h
NXshadowEvent.h and Regions.h.
- Modified the Makefile.in file and configure scripts in order to
compile the component.
nxaccess-3.0.0-13
- Fixed X11Poller.cpp pre-processing.
nxaccess-3.0.0-12
- Fixed build problems on Solaris.
nxaccess-3.0.0-11
- Added NXShadowUpdateBuffer() function. This function creates the
buffer for the polling.
- If the scaline fails, the polling will suspend for 50 ms.
- Added some functions in order to access data member class.
nxaccess-3.0.0-10
- Used XTEST extension to make the shared display create input events.
nxaccess-3.0.0-9
- Added the mouse events.
- Now, it's possible to connect to X server display or agent display,
the display's content is captured by GetImage and sent to another
display by PutImage.
nxaccess-3.0.0-8
- Added KeysCursorTmp.cpp file.
- Solved a problem with the keys, when the window lost focus the Key-
Release events weren't propagated.
nxaccess-3.0.0-7
- Added functions in order to remove issues with some keys combina-
tion.
nxaccess-3.0.0-6
- Added functions to simulate keys Ctrl + Alt + Del in order to run
the Task Manager application.
- Now nxaccess is able to manage all switches between desktops.
nxaccess-3.0.0-5
- Solved a problem with the cursor shape. After a while, the cursor
shape are no more updated.
- Now the cursor is updated only when it changes its shape.
- Removed a dirty lines after screen updates.
- Removed the unused file Keyboard.cpp.
- Added the colorCorrect() macro in NXaccess.h.
- Removed the colorCorrect() function in Updater.cpp.
nxaccess-3.0.0-4
- Renamed some source files and functions conforming them to the name
of component.
nxaccess-3.0.0-3
- Removed the parameter of type Display in all methods of the class
Poller.
- Print, Break and Pause keys are enabled.
nxaccess-3.0.0-2
- Modified the Makefile.in in order to avoid compiling the executive
file.
- Removed the unused file Main.cpp.
- The Windows keys are enabled.
- Synchronized local and remote states of Num_Lock, Caps_Lock and
Scroll_Lock.
- Updated the NoMachine copyright notices.
nxaccess-3.0.0-1
- Opened the 3.0.0 branch based on the nxmirror-2.0.0-3.
nxmirror-2.0.0-3
- Added the keyboard events for all layouts.
- The mouse wheel button is enabled.
nxmirror-2.0.0-2
- Completed implementation of the keyboard events only for italian
layout.
- Added the mouse events and shape cursor.
nxmirror-2.0.0-1
- Opened the 2.0.0 branch based on the 1.5.0-60.
- Added CHANGELOG.
GNU GENERAL PUBLIC LICENSE
Version 2, June 1991
Copyright (C) 1989, 1991 Free Software Foundation, Inc.,
51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
Everyone is permitted to copy and distribute verbatim copies
of this license document, but changing it is not allowed.
Preamble
The licenses for most software are designed to take away your
freedom to share and change it. By contrast, the GNU General Public
License is intended to guarantee your freedom to share and change free
software--to make sure the software is free for all its users. This
General Public License applies to most of the Free Software
Foundation's software and to any other program whose authors commit to
using it. (Some other Free Software Foundation software is covered by
the GNU Lesser General Public License instead.) You can apply it to
your programs, too.
When we speak of free software, we are referring to freedom, not
price. Our General Public Licenses are designed to make sure that you
have the freedom to distribute copies of free software (and charge for
this service if you wish), that you receive source code or can get it
if you want it, that you can change the software or use pieces of it
in new free programs; and that you know you can do these things.
To protect your rights, we need to make restrictions that forbid
anyone to deny you these rights or to ask you to surrender the rights.
These restrictions translate to certain responsibilities for you if you
distribute copies of the software, or if you modify it.
For example, if you distribute copies of such a program, whether
gratis or for a fee, you must give the recipients all the rights that
you have. You must make sure that they, too, receive or can get the
source code. And you must show them these terms so they know their
rights.
We protect your rights with two steps: (1) copyright the software, and
(2) offer you this license which gives you legal permission to copy,
distribute and/or modify the software.
Also, for each author's protection and ours, we want to make certain
that everyone understands that there is no warranty for this free
software. If the software is modified by someone else and passed on, we
want its recipients to know that what they have is not the original, so
that any problems introduced by others will not reflect on the original
authors' reputations.
Finally, any free program is threatened constantly by software
patents. We wish to avoid the danger that redistributors of a free
program will individually obtain patent licenses, in effect making the
program proprietary. To prevent this, we have made it clear that any
patent must be licensed for everyone's free use or not licensed at all.
The precise terms and conditions for copying, distribution and
modification follow.
GNU GENERAL PUBLIC LICENSE
TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION
0. This License applies to any program or other work which contains
a notice placed by the copyright holder saying it may be distributed
under the terms of this General Public License. The "Program", below,
refers to any such program or work, and a "work based on the Program"
means either the Program or any derivative work under copyright law:
that is to say, a work containing the Program or a portion of it,
either verbatim or with modifications and/or translated into another
language. (Hereinafter, translation is included without limitation in
the term "modification".) Each licensee is addressed as "you".
Activities other than copying, distribution and modification are not
covered by this License; they are outside its scope. The act of
running the Program is not restricted, and the output from the Program
is covered only if its contents constitute a work based on the
Program (independent of having been made by running the Program).
Whether that is true depends on what the Program does.
1. You may copy and distribute verbatim copies of the Program's
source code as you receive it, in any medium, provided that you
conspicuously and appropriately publish on each copy an appropriate
copyright notice and disclaimer of warranty; keep intact all the
notices that refer to this License and to the absence of any warranty;
and give any other recipients of the Program a copy of this License
along with the Program.
You may charge a fee for the physical act of transferring a copy, and
you may at your option offer warranty protection in exchange for a fee.
2. You may modify your copy or copies of the Program or any portion
of it, thus forming a work based on the Program, and copy and
distribute such modifications or work under the terms of Section 1
above, provided that you also meet all of these conditions:
a) You must cause the modified files to carry prominent notices
stating that you changed the files and the date of any change.
b) You must cause any work that you distribute or publish, that in
whole or in part contains or is derived from the Program or any
part thereof, to be licensed as a whole at no charge to all third
parties under the terms of this License.
c) If the modified program normally reads commands interactively
when run, you must cause it, when started running for such
interactive use in the most ordinary way, to print or display an
announcement including an appropriate copyright notice and a
notice that there is no warranty (or else, saying that you provide
a warranty) and that users may redistribute the program under
these conditions, and telling the user how to view a copy of this
License. (Exception: if the Program itself is interactive but
does not normally print such an announcement, your work based on
the Program is not required to print an announcement.)
These requirements apply to the modified work as a whole. If
identifiable sections of that work are not derived from the Program,
and can be reasonably considered independent and separate works in
themselves, then this License, and its terms, do not apply to those
sections when you distribute them as separate works. But when you
distribute the same sections as part of a whole which is a work based
on the Program, the distribution of the whole must be on the terms of
this License, whose permissions for other licensees extend to the
entire whole, and thus to each and every part regardless of who wrote it.
Thus, it is not the intent of this section to claim rights or contest
your rights to work written entirely by you; rather, the intent is to
exercise the right to control the distribution of derivative or
collective works based on the Program.
In addition, mere aggregation of another work not based on the Program
with the Program (or with a work based on the Program) on a volume of
a storage or distribution medium does not bring the other work under
the scope of this License.
3. You may copy and distribute the Program (or a work based on it,
under Section 2) in object code or executable form under the terms of
Sections 1 and 2 above provided that you also do one of the following:
a) Accompany it with the complete corresponding machine-readable
source code, which must be distributed under the terms of Sections
1 and 2 above on a medium customarily used for software interchange; or,
b) Accompany it with a written offer, valid for at least three
years, to give any third party, for a charge no more than your
cost of physically performing source distribution, a complete
machine-readable copy of the corresponding source code, to be
distributed under the terms of Sections 1 and 2 above on a medium
customarily used for software interchange; or,
c) Accompany it with the information you received as to the offer
to distribute corresponding source code. (This alternative is
allowed only for noncommercial distribution and only if you
received the program in object code or executable form with such
an offer, in accord with Subsection b above.)
The source code for a work means the preferred form of the work for
making modifications to it. For an executable work, complete source
code means all the source code for all modules it contains, plus any
associated interface definition files, plus the scripts used to
control compilation and installation of the executable. However, as a
special exception, the source code distributed need not include
anything that is normally distributed (in either source or binary
form) with the major components (compiler, kernel, and so on) of the
operating system on which the executable runs, unless that component
itself accompanies the executable.
If distribution of executable or object code is made by offering
access to copy from a designated place, then offering equivalent
access to copy the source code from the same place counts as
distribution of the source code, even though third parties are not
compelled to copy the source along with the object code.
4. You may not copy, modify, sublicense, or distribute the Program
except as expressly provided under this License. Any attempt
otherwise to copy, modify, sublicense or distribute the Program is
void, and will automatically terminate your rights under this License.
However, parties who have received copies, or rights, from you under
this License will not have their licenses terminated so long as such
parties remain in full compliance.
5. You are not required to accept this License, since you have not
signed it. However, nothing else grants you permission to modify or
distribute the Program or its derivative works. These actions are
prohibited by law if you do not accept this License. Therefore, by
modifying or distributing the Program (or any work based on the
Program), you indicate your acceptance of this License to do so, and
all its terms and conditions for copying, distributing or modifying
the Program or works based on it.
6. Each time you redistribute the Program (or any work based on the
Program), the recipient automatically receives a license from the
original licensor to copy, distribute or modify the Program subject to
these terms and conditions. You may not impose any further
restrictions on the recipients' exercise of the rights granted herein.
You are not responsible for enforcing compliance by third parties to
this License.
7. If, as a consequence of a court judgment or allegation of patent
infringement or for any other reason (not limited to patent issues),
conditions are imposed on you (whether by court order, agreement or
otherwise) that contradict the conditions of this License, they do not
excuse you from the conditions of this License. If you cannot
distribute so as to satisfy simultaneously your obligations under this
License and any other pertinent obligations, then as a consequence you
may not distribute the Program at all. For example, if a patent
license would not permit royalty-free redistribution of the Program by
all those who receive copies directly or indirectly through you, then
the only way you could satisfy both it and this License would be to
refrain entirely from distribution of the Program.
If any portion of this section is held invalid or unenforceable under
any particular circumstance, the balance of the section is intended to
apply and the section as a whole is intended to apply in other
circumstances.
It is not the purpose of this section to induce you to infringe any
patents or other property right claims or to contest validity of any
such claims; this section has the sole purpose of protecting the
integrity of the free software distribution system, which is
implemented by public license practices. Many people have made
generous contributions to the wide range of software distributed
through that system in reliance on consistent application of that
system; it is up to the author/donor to decide if he or she is willing
to distribute software through any other system and a licensee cannot
impose that choice.
This section is intended to make thoroughly clear what is believed to
be a consequence of the rest of this License.
8. If the distribution and/or use of the Program is restricted in
certain countries either by patents or by copyrighted interfaces, the
original copyright holder who places the Program under this License
may add an explicit geographical distribution limitation excluding
those countries, so that distribution is permitted only in or among
countries not thus excluded. In such case, this License incorporates
the limitation as if written in the body of this License.
9. The Free Software Foundation may publish revised and/or new versions
of the General Public License from time to time. Such new versions will
be similar in spirit to the present version, but may differ in detail to
address new problems or concerns.
Each version is given a distinguishing version number. If the Program
specifies a version number of this License which applies to it and "any
later version", you have the option of following the terms and conditions
either of that version or of any later version published by the Free
Software Foundation. If the Program does not specify a version number of
this License, you may choose any version ever published by the Free Software
Foundation.
10. If you wish to incorporate parts of the Program into other free
programs whose distribution conditions are different, write to the author
to ask for permission. For software which is copyrighted by the Free
Software Foundation, write to the Free Software Foundation; we sometimes
make exceptions for this. Our decision will be guided by the two goals
of preserving the free status of all derivatives of our free software and
of promoting the sharing and reuse of software generally.
NO WARRANTY
11. BECAUSE THE PROGRAM IS LICENSED FREE OF CHARGE, THERE IS NO WARRANTY
FOR THE PROGRAM, TO THE EXTENT PERMITTED BY APPLICABLE LAW. EXCEPT WHEN
OTHERWISE STATED IN WRITING THE COPYRIGHT HOLDERS AND/OR OTHER PARTIES
PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESSED
OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE. THE ENTIRE RISK AS
TO THE QUALITY AND PERFORMANCE OF THE PROGRAM IS WITH YOU. SHOULD THE
PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF ALL NECESSARY SERVICING,
REPAIR OR CORRECTION.
12. IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING
WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MAY MODIFY AND/OR
REDISTRIBUTE THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES,
INCLUDING ANY GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING
OUT OF THE USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED
TO LOSS OF DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY
YOU OR THIRD PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER
PROGRAMS), EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE
POSSIBILITY OF SUCH DAMAGES.
END OF TERMS AND CONDITIONS
How to Apply These Terms to Your New Programs
If you develop a new program, and you want it to be of the greatest
possible use to the public, the best way to achieve this is to make it
free software which everyone can redistribute and change under these terms.
To do so, attach the following notices to the program. It is safest
to attach them to the start of each source file to most effectively
convey the exclusion of warranty; and each file should have at least
the "copyright" line and a pointer to where the full notice is found.
<one line to give the program's name and a brief idea of what it does.>
Copyright (C) <year> <name of author>
This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 2 of the License, or
(at your option) any later version.
This program 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 for more details.
You should have received a copy of the GNU General Public License along
with this program; if not, write to the Free Software Foundation, Inc.,
51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
Also add information on how to contact you by electronic and paper mail.
If the program is interactive, make it output a short notice like this
when it starts in an interactive mode:
Gnomovision version 69, Copyright (C) year name of author
Gnomovision comes with ABSOLUTELY NO WARRANTY; for details type `show w'.
This is free software, and you are welcome to redistribute it
under certain conditions; type `show c' for details.
The hypothetical commands `show w' and `show c' should show the appropriate
parts of the General Public License. Of course, the commands you use may
be called something other than `show w' and `show c'; they could even be
mouse-clicks or menu items--whatever suits your program.
You should also get your employer (if you work as a programmer) or your
school, if any, to sign a "copyright disclaimer" for the program, if
necessary. Here is a sample; alter the names:
Yoyodyne, Inc., hereby disclaims all copyright interest in the program
`Gnomovision' (which makes passes at compilers) written by James Hacker.
<signature of Ty Coon>, 1 April 1989
Ty Coon, President of Vice
This General Public License does not permit incorporating your program into
proprietary programs. If your program is a subroutine library, you may
consider it more useful to permit linking proprietary applications with the
library. If this is what you want to do, use the GNU Lesser General
Public License instead of this License.
/**************************************************************************/
/* */
/* Copyright (c) 2001, 2007 NoMachine, http://www.nomachine.com/. */
/* */
/* NXCOMPSHAD, NX protocol compression and NX extensions to this software */
/* are copyright of NoMachine. Redistribution and use of the present */
/* software is allowed according to terms specified in the file LICENSE */
/* which comes in the source distribution. */
/* */
/* Check http://www.nomachine.com/licensing.html for applicability. */
/* */
/* NX and NoMachine are trademarks of Medialogic S.p.A. */
/* */
/* All rights reserved. */
/* */
/**************************************************************************/
#include <string.h>
#include <sys/time.h>
#define PANIC
#define WARNING
#undef TEST
#undef DEBUG
#include "Core.h"
#include "Logger.h"
const int CorePoller::maxSliceHeight_ = 20;
const int CorePoller::minSliceHeight_ = 3;
const char CorePoller::interlace_[] =
{
0, 16,
8, 24,
4, 20, 12, 28,
2, 18, 10, 26, 6, 22, 14, 30,
1, 17,
9, 25,
5, 21, 13, 29,
3, 19, 11, 27, 7, 23, 15, 31
};
CorePoller::CorePoller(Input *input, Display *display) : input_(input)
{
logTrace("CorePoller::CorePoller");
buffer_ = NULL;
lastUpdatedRegion_ = NULL;
lineStatus_ = NULL;
linePriority_ = NULL;
lefts_ = NULL;
rights_ = NULL;
}
CorePoller::~CorePoller()
{
logTrace("CorePoller::~CorePoller");
if (buffer_ != NULL)
{
delete [] buffer_;
buffer_ = NULL;
}
if (lastUpdatedRegion_ != NULL)
{
XDestroyRegion(lastUpdatedRegion_);
lastUpdatedRegion_ = NULL;
}
if (lineStatus_ != NULL)
{
delete [] lineStatus_;
lineStatus_ = NULL;
}
if (linePriority_ != NULL)
{
delete [] linePriority_;
linePriority_ = NULL;
}
if (lefts_ != NULL)
{
delete [] lefts_;
lefts_ = NULL;
}
if (rights_ != NULL)
{
delete [] rights_;
rights_ = NULL;
}
}
int CorePoller::init()
{
logTrace("CorePoller::init");
createFrameBuffer();
if (buffer_ == NULL)
{
logError("CorePoller::init", ESET(ENOMEM));
return -1;
}
logTest("CorePoller::init", "Allocated frame buffer at [%p] for [%d] bytes.",
buffer_, bpl_ * height_);
if (lastUpdatedRegion_ != NULL)
{
XDestroyRegion(lastUpdatedRegion_);
lastUpdatedRegion_ = NULL;
}
lastUpdatedRegion_ = XCreateRegion();
if (lineStatus_ != NULL)
{
delete[] lineStatus_;
}
lineStatus_ = new LineStatus[height_ + 1];
if (lineStatus_ == NULL)
{
logError("CorePoller::init", ESET(ENOMEM));
return -1;
}
//
// We need this boundary element to
// speed up the algo.
//
if (linePriority_ != NULL)
{
delete[] linePriority_;
}
linePriority_ = new int [height_ + 1];
if (linePriority_ == NULL)
{
logError("CorePoller::init", ESET(ENOMEM));
return -1;
}
for (unsigned int i = 0; i < height_; i++)
{
linePriority_[i] = HIGHEST_PRIORITY;
}
if (lefts_ != NULL)
{
delete[] lefts_;
}
lefts_ = new int [height_];
if (rights_ != NULL)
{
delete[] rights_;
}
rights_ = new int [height_];
for (unsigned int i = 0; i < height_; i++)
{
rights_[i] = lefts_[i] = 0;
}
return 1;
}
int CorePoller::isChanged(int (*checkIfInputCallback)(void *), void *arg, int *suspended)
{
logTrace("CorePoller::isChanged");
#if defined(__CYGWIN32__) || defined(WIN32)
checkDesktop();
#endif
#if !defined(__CYGWIN32__) && !defined(WIN32)
if (mirror_ == 1)
{
int result = mirrorChanges_;
mirrorChanges_ = 0;
return result;
}
#endif
logDebug("CorePoller:isChanged", "Going to use default polling algorithm.\n");
//
// In order to allow this function to
// be suspended and resumed later, we
// need to save these two status vars.
//
static int idxIlace = 0;
static int curLine = 0;
const long timeout = 50;
long oldTime;
long newTime;
struct timeval ts;
gettimeofday(&ts, NULL);
oldTime = ts.tv_sec * 1000 + ts.tv_usec / 1000;
if (curLine == 0) // && idxIlace == 0 ?
{
for (unsigned int i = 0; i < height_; i++)
{
lineStatus_[i] = LINE_NOT_CHECKED;
}
}
int foundChanges = 0;
foundChanges = 0;
int curIlace = interlace_[idxIlace];
bool moveBackward = false;
logDebug("CorePoller::isChanged", "Interlace index [%d] interlace [%d].", idxIlace, curIlace);
for (; curLine < (int) height_; curLine++)
{
logDebug("CorePoller::isChanged", "Analizing line [%d] move backward [%d] status [%d] priority [%d].",
curLine, moveBackward, lineStatus_[curIlace], linePriority_[curLine]);
//
// Ask the caller if the polling have to be suspended.
//
if ((*checkIfInputCallback)(arg) == 1)
{
*suspended = 1;
break;
}
//
// Suspend if too much time is elapsed.
//
gettimeofday(&ts, NULL);
newTime = ts.tv_sec * 1000 + ts.tv_usec / 1000;
if (newTime - oldTime >= timeout)
{
*suspended = 1;
break;
}
oldTime = newTime;
if (lineStatus_[curLine] != LINE_NOT_CHECKED)
{
continue;
}
if (moveBackward)
{
moveBackward = false;
}
else
{
switch (linePriority_[curLine])
{
case 1:
case 29:
{
//
// It was a priority,
// but now it may not be.
//
}
case 31:
{
//
// Not a priority, still isn't.
//
linePriority_[curLine] = NOT_PRIORITY;
break;
}
case 0:
{
//
// Make it a priority.
//
linePriority_[curLine] = PRIORITY;
break;
}
default:
{
linePriority_[curLine]--;
break;
}
}
if ((linePriority_[curLine] > PRIORITY) && ((curLine & 31) != curIlace))
{
continue;
}
}
XRectangle rect = {0, curLine, width_, 1};
char *buffer;
logDebug("CorePoller::isChanged", "Checking line [%d].", curLine);
if ((buffer = getRect(rect)) == NULL)
{
logDebug("CorePoller::isChanged", "Failed to retrieve line [%d].", curLine);
return -1;
}
if (memcmp(buffer, buffer_ + curLine * bpl_, bpl_) == 0 || differ(buffer, rect) == 0)
{
logDebug("CorePoller::isChanged", "Data buffer didn't change.");
lineStatus_[curLine] = LINE_NOT_CHANGED;
continue;
}
rect.x = lefts_[rect.y];
rect.width = rights_[rect.y] - lefts_[rect.y] + 1;
update(buffer + rect.x * bpp_, rect);
foundChanges = 1;
lineStatus_[curLine] = LINE_HAS_CHANGED;
//
// Wake up the next line.
//
if (linePriority_[curLine + 1] > PRIORITY)
{
linePriority_[curLine + 1] = HIGHEST_PRIORITY;
}
//
// Give this line priority.
//
linePriority_[curLine] = HIGHEST_PRIORITY;
//
// Wake up previous line.
//
if (curLine > 0 && lineStatus_[curLine - 1] == LINE_NOT_CHECKED)
{
moveBackward = true;
curLine -= 2;
}
}
//
// Execution reached the end of loop.
//
if (curLine == (int) height_)
{
idxIlace = (idxIlace + 1) % 32;
curLine = 0;
}
//
// Create the region of changed pixels.
//
if (foundChanges)
{
int start, last, curLine, left, right;
for (curLine = 0; curLine < (int) height_; curLine++)
{
if (lineStatus_[curLine] == LINE_HAS_CHANGED)
{
break;
}
}
start = curLine;
last = curLine;
left = lefts_[curLine];
right = rights_[curLine];
curLine++;
while (1)
{
for (; curLine < (int) height_; curLine++)
{
if (lineStatus_[curLine] == LINE_HAS_CHANGED)
{
break;
}
}
if (curLine == (int) height_)
{
break;
}
if ((curLine - last > minSliceHeight_) || (last - start > maxSliceHeight_))
{
XRectangle rect = {left, start, right - left + 1, last - start + 1};
XUnionRectWithRegion(&rect, lastUpdatedRegion_, lastUpdatedRegion_);
start = curLine;
left = lefts_[curLine];
right = rights_[curLine];
}
else
{
if (lefts_[curLine] < left)
{
left = lefts_[curLine];
}
if (rights_[curLine] > right)
{
right = rights_[curLine];
}
}
last = curLine;
curLine++;
}
//
// Send last block.
//
if (last >= start)
{
XRectangle rect = {left, start, right - left + 1, last - start + 1};
XUnionRectWithRegion(&rect, lastUpdatedRegion_, lastUpdatedRegion_);
}
}
return foundChanges;
}
int CorePoller::differ(char *buffer, XRectangle r)
{
logTrace("CorePoller::differ");
int bpl = bpp_ * r.width;
int i;
char *pBuf;
char *pFb;
pBuf = (buffer);
pFb = (buffer_ + r.x + r.y * bpl_);
for (i = 0; i < bpl; i++)
{
if (*pFb++ != *pBuf++)
{
lefts_[r.y] = i / bpp_;
break;
}
}
if (i == bpl)
{
return 0;
}
pBuf = (buffer) + bpl - 1;
pFb = (buffer_ + r.x + r.y * bpl_) + bpl - 1;
int j = i - 1;
for (i = bpl - 1; i > j; i--)
{
if (*pFb-- != *pBuf--)
{
rights_[r.y] = i / bpp_;
break;
}
}
return 1;
}
void CorePoller::update(char *src, XRectangle r)
{
logTrace("CorePoller::update");
char *dst = buffer_ + r.x * bpp_ + r.y * bpl_;
int bpl = bpp_ * r.width;
for (unsigned int i = 0; i < r.height; i++)
{
memcpy(dst, src, bpl);
src += bpl;
dst += bpl_;
}
}
void CorePoller::handleEvent(Display *display, XEvent *event)
{
logTrace("CorePoller::handleEvent");
switch (event -> type)
{
case KeyPress:
case KeyRelease:
{
handleKeyboardEvent(display, event);
break;
}
case ButtonPress:
case ButtonRelease:
case MotionNotify:
{
handleMouseEvent(display, event);
break;
}
default:
{
logTest("CorePoller::handleEvent", "Handling unexpected event [%d] from display [%p].",
event -> type, display);
break;
}
}
}
void CorePoller::handleInput()
{
while (input_ -> checkIfEvent())
{
Display *display = input_ -> currentDisplay();
XEvent *event = input_ -> popEvent();
handleEvent(display, event);
delete event;
}
}
void CorePoller::createFrameBuffer()
{
logTrace("CorePoller::createFrameBuffer");
if (buffer_ == NULL)
{
buffer_ = new char[bpl_ * height_];
}
}
/**************************************************************************/
/* */
/* Copyright (c) 2001, 2007 NoMachine, http://www.nomachine.com/. */
/* */
/* NXCOMPSHAD, NX protocol compression and NX extensions to this software */
/* are copyright of NoMachine. Redistribution and use of the present */
/* software is allowed according to terms specified in the file LICENSE */
/* which comes in the source distribution. */
/* */
/* Check http://www.nomachine.com/licensing.html for applicability. */
/* */
/* NX and NoMachine are trademarks of Medialogic S.p.A. */
/* */
/* All rights reserved. */
/* */
/**************************************************************************/
#ifndef CorePoller_H
#define CorePoller_H
#include <stdio.h>
#include "Logger.h"
#include "Regions.h"
#include "Input.h"
typedef enum{
LINE_HAS_CHANGED,
LINE_NOT_CHECKED,
LINE_NOT_CHANGED
} LineStatus;
typedef enum{
HIGHEST_PRIORITY = 0,
PRIORITY = 30,
NOT_PRIORITY = 90
} LinePriority;
class CorePoller
{
public:
CorePoller(Input*, Display*);
virtual ~CorePoller();
virtual int init();
unsigned int width() const;
unsigned int height() const;
unsigned char depth() const;
int isChanged(int (*)(void*), void *, int *);
char *getFrameBuffer() const;
void destroyFrameBuffer();
void createFrameBuffer();
Region lastUpdatedRegion();
Region getLastUpdatedRegion();
void handleInput();
void handleEvent(Display *, XEvent *);
Display *getShadowDisplay();
void setShadowDisplay(Display *shadowDisplay);
protected:
unsigned int bpp_;
unsigned int bpl_;
unsigned int width_;
unsigned int height_;
int depth_;
char *buffer_;
unsigned long redMask_;
unsigned long greenMask_;
unsigned long blueMask_;
unsigned long colorMask_[3];
char mirror_;
char mirrorChanges_;
virtual int updateShadowFrameBuffer(void) = 0;
virtual char *getRect(XRectangle r) = 0;
int imageByteOrder_;
#ifdef __CYGWIN32__
virtual char checkDesktop(void) = 0;
#endif
Display *shadowDisplay_;
void update(char *src, XRectangle r);
Region lastUpdatedRegion_;
private:
virtual void handleKeyboardEvent(Display *, XEvent *) = 0;
virtual void handleMouseEvent(Display *, XEvent *) = 0;
Input *input_;
static const int maxSliceHeight_;
static const int minSliceHeight_;
LineStatus *lineStatus_;
int *linePriority_;
static const char interlace_[];
int *lefts_;
int *rights_;
// FIXME: Make them friend.
int differ(char *src, XRectangle r);
};
inline unsigned int CorePoller::width() const
{
return width_;
}
inline unsigned int CorePoller::height() const
{
return height_;
}
inline unsigned char CorePoller::depth() const
{
return depth_;
}
inline char *CorePoller::getFrameBuffer() const
{
return buffer_;
}
inline void CorePoller::destroyFrameBuffer()
{
if (buffer_ != NULL)
{
delete[] buffer_;
buffer_ = NULL;
}
}
inline Region CorePoller::lastUpdatedRegion()
{
Region region = lastUpdatedRegion_;
lastUpdatedRegion_ = XCreateRegion();
if (lastUpdatedRegion_ == NULL)
{
logError("CorePoller::lastUpdatedRegion", ESET(ENOMEM));
lastUpdatedRegion_ = region;
return NULL;
}
return region;
}
inline Region CorePoller::getLastUpdatedRegion()
{
return lastUpdatedRegion_;
}
inline Display *CorePoller::getShadowDisplay()
{
return shadowDisplay_ ;
}
inline void CorePoller::setShadowDisplay(Display *shadowDisplay)
{
shadowDisplay_ = shadowDisplay;
}
#endif /* CorePoller_H */
/**************************************************************************/
/* */
/* Copyright (c) 2001, 2007 NoMachine, http://www.nomachine.com/. */
/* */
/* NXCOMPSHAD, NX protocol compression and NX extensions to this software */
/* are copyright of NoMachine. Redistribution and use of the present */
/* software is allowed according to terms specified in the file LICENSE */
/* which comes in the source distribution. */
/* */
/* Check http://www.nomachine.com/licensing.html for applicability. */
/* */
/* NX and NoMachine are trademarks of Medialogic S.p.A. */
/* */
/* All rights reserved. */
/* */
/**************************************************************************/
#include <string.h>
#define PANIC
#define WARNING
#undef TEST
#undef DEBUG
#include "Input.h"
#include "Logger.h"
Input::Input()
{
logTrace("Input::Input");
eventsHead_ = NULL;
eventsTail_ = NULL;
keymap_ = NULL;
}
Input::~Input()
{
logTrace("Input::~Input");
Event *head = eventsHead_;
while (head)
{
Event *next = head -> next;
delete head -> event;
delete head;
head = next;
}
if (keymap_ != NULL)
{
logDebug("Input::~Input", "Delete keymap_ [%p].", keymap_);
delete [] keymap_;
}
}
void Input::pushEvent(Display *display, XEvent *event)
{
Event *tail = new Event;
if (tail == NULL)
{
logError("Input::pushEvent", ESET(ENOMEM));
return;
}
tail -> next = NULL;
tail -> display = display;
tail -> event = event;
if (eventsHead_ == NULL)
{
eventsHead_ = tail;
}
else
{
eventsTail_ -> next = tail;
}
eventsTail_ = tail;
}
XEvent *Input::popEvent()
{
Event *head = eventsHead_;
if (head == NULL)
{
return 0;
}
XEvent *event = head -> event;
eventsHead_ = head -> next;
delete head;
if (eventsHead_ == NULL)
{
eventsTail_ = NULL;
}
return event;
}
int Input::removeAllEvents(Display *display)
{
logTrace("Input::removeAllEvents");
int nRemoved = 0;
Event *current = eventsHead_;
while (current)
{
if (display == current -> display)
{
//
// Update head of list.
//
if (current == eventsHead_)
{
eventsHead_ = current -> next;
}
//
// Update tail of list.
//
if (current == eventsTail_)
{
eventsTail_ = eventsHead_;
while (eventsTail_ && eventsTail_ -> next)
{
eventsTail_ = eventsTail_ -> next;
}
}
//
// Remove event.
//
Event *next = current -> next;
delete current -> event;
delete current;
current = next;
nRemoved++;
}
else
{
current = current -> next;
}
}
return nRemoved;
}
/**************************************************************************/
/* */
/* Copyright (c) 2001, 2007 NoMachine, http://www.nomachine.com/. */
/* */
/* NXCOMPSHAD, NX protocol compression and NX extensions to this software */
/* are copyright of NoMachine. Redistribution and use of the present */
/* software is allowed according to terms specified in the file LICENSE */
/* which comes in the source distribution. */
/* */
/* Check http://www.nomachine.com/licensing.html for applicability. */
/* */
/* NX and NoMachine are trademarks of Medialogic S.p.A. */
/* */
/* All rights reserved. */
/* */
/**************************************************************************/
#ifndef Input_H
#define Input_H
#include <X11/Xlib.h>
typedef struct Event
{
struct Event *next;
Display *display;
XEvent *event;
} Event;
class Input
{
public:
Input();
~Input();
int checkIfEvent();
void pushEvent(Display *, XEvent *);
XEvent *popEvent();
Display *currentDisplay();
int removeAllEvents(Display *);
void setKeymap(char *keymap);
char *getKeymap();
void setShadowDisplayName(char *shadowDisplayName);
char *getShadowDisplayName();
private:
Event *eventsHead_;
Event *eventsTail_;
char *keymap_;
char *shadowDisplayName_;
};
inline Display *Input::currentDisplay()
{
return eventsHead_ ? eventsHead_ -> display : NULL;
}
inline int Input::checkIfEvent()
{
return (eventsHead_ != NULL);
}
inline void Input::setKeymap(char *keymap)
{
keymap_ = keymap;
}
inline char *Input::getKeymap()
{
return keymap_;
}
inline void Input::setShadowDisplayName(char *shadowDisplayName)
{
shadowDisplayName_ = shadowDisplayName;
}
inline char *Input::getShadowDisplayName()
{
return shadowDisplayName_;
}
#endif /* Input_H */
Copyright (C) 2001, 2007 NoMachine - http://www.nomachine.com/.
NXCOMPSHAD and NX extensions to X are copyright of NoMachine.
Redistribution and use of this software is allowed according to the
following terms:
This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License Version 2, and
not any other version, as published by the Free Software Foundation.
This program is distributed in the hope that it will be useful, but
WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTA-
BILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General
Public License for more details.
You should have received a copy of the GNU General Public License
along with this program; if not, you can request a copy to NoMachine
or write to the Free Software Foundation, Inc., 59 Temple Place,
Suite 330, Boston, MA 02111-1307 USA
All rights reserved.
/**************************************************************************/
/* */
/* Copyright (c) 2001, 2007 NoMachine, http://www.nomachine.com. */
/* */
/* NXCOMP, NX protocol compression and NX extensions to this software */
/* are copyright of NoMachine. Redistribution and use of the present */
/* software is allowed according to terms specified in the file LICENSE */
/* which comes in the source distribution. */
/* */
/* Check http://www.nomachine.com/licensing.html for applicability. */
/* */
/* NX and NoMachine are trademarks of Medialogic S.p.A. */
/* */
/* All rights reserved. */
/* */
/**************************************************************************/
#include <stdio.h>
#include <ctype.h>
#include <unistd.h>
#define PANIC
#define WARNING
#undef TEST
#define DEBUG
#include "Misc.h"
#include "Logger.h"
Logger logger;
void Logger::user(const char *format, va_list arguments)
{
char string[1024];
vsnprintf(string, 1024, format, arguments);
fprintf(stderr, "%s\n", string);
}
void Logger::error(const char *name, int error)
{
fprintf(stderr, "PANIC! %s: Failed with code %d: %s\n",
name, error, strerror(error));
}
void Logger::warning(const char *name, const char *format, va_list arguments)
{
char string[1024];
vsnprintf(string, 1024, format, arguments);
fprintf(stderr, "%s: WARNING! %s\n", name, string);
}
void Logger::test(const char *name, const char *format, va_list arguments)
{
char string[1024];
vsnprintf(string, 1024, format, arguments);
fprintf(stderr, "%s: %s\n", name, string);
}
void Logger::trace(const char *name)
{
fprintf(stderr, "%s\n", name);
}
void Logger::debug(const char *name, const char *format, va_list arguments)
{
char string[1024];
vsnprintf(string, 1024, format, arguments);
fprintf(stderr, "%s: %s\n", name, string);
}
void Logger::dump(const char *name, const char *data, int size)
{
fprintf(stderr, "%s: Dumping %d bytes of data at %p\n",
name, size, data);
for (int i = 0; i < size;)
{
fprintf(stderr, "[%d]\t", i);
int t = i;
for (unsigned int ii = 0; i < size && ii < 8; i++, ii++)
{
fprintf(stderr, "%02x/%d\t", data[i] & 0xff, data[i]);
}
for (unsigned int ii = i % 8; ii > 0 && ii < 8; ii++)
{
fprintf(stderr, "\t");
}
i = t;
for (unsigned int ii = 0; i < size && ii < 8; i++, ii++)
{
if (isprint(data[i]))
{
fprintf(stderr, "%c", data[i]);
}
else
{
fprintf(stderr, ".");
}
}
fprintf(stderr, "\n");
}
}
/**************************************************************************/
/* */
/* Copyright (c) 2001, 2007 NoMachine, http://www.nomachine.com. */
/* */
/* NXCOMP, NX protocol compression and NX extensions to this software */
/* are copyright of NoMachine. Redistribution and use of the present */
/* software is allowed according to terms specified in the file LICENSE */
/* which comes in the source distribution. */
/* */
/* Check http://www.nomachine.com/licensing.html for applicability. */
/* */
/* NX and NoMachine are trademarks of Medialogic S.p.A. */
/* */
/* All rights reserved. */
/* */
/**************************************************************************/
#ifndef Logger_H
#define Logger_H
#include <errno.h>
#include <stdarg.h>
//
// Error handling macros.
//
#define ESET(e) (errno = (e))
#define EGET() (errno)
#define ESTR() strerror(errno)
extern class Logger logger;
class Logger
{
public:
void user(const char *format, va_list arguments);
void error(const char *name, int error);
void warning(const char *name, const char *format, va_list arguments);
void test(const char *name, const char *format, va_list arguments);
void trace(const char *name);
void debug(const char *name, const char *format, va_list arguments);
void dump(const char *name, const char *data, int size);
};
static inline void logUser(const char *format, ...) \
__attribute__((format(printf, 1, 2))) __attribute__((__unused__));
static inline void logError(const char *name, int error) \
__attribute__((__unused__));
static inline void logWarning(const char *name, const char *format, ...) \
__attribute__((__unused__));
static inline void logTest(const char *name, const char *format, ...) \
__attribute__((format(printf, 2, 3))) __attribute__((__unused__));
static inline void logTrace(const char *name) \
__attribute__((__unused__));
static inline void logDebug(const char *name, const char *format, ...) \
__attribute__((format(printf, 2, 3))) __attribute__((__unused__));
static inline void logDump(const char *name, const char *data, int size) \
__attribute__((__unused__));
static inline void logUser(const char *format, ...)
{
va_list arguments;
va_start(arguments, format);
logger.user(format, arguments);
va_end(arguments);
}
static inline void logError(const char *name, int error)
{
#if defined(DEBUG) || defined(TEST) || \
defined(WARNING) || defined(PANIC)
logger.error(name, error);
#endif
}
static inline void logWarning(const char *name, const char *format, ...)
{
#if defined(DEBUG) || defined(TEST) || \
defined(WARNING)
va_list arguments;
va_start(arguments, format);
logger.warning(name, format, arguments);
va_end(arguments);
#endif
}
static inline void logTest(const char *name, const char *format, ...)
{
#if defined(TEST)
va_list arguments;
va_start(arguments, format);
logger.test(name, format, arguments);
va_end(arguments);
#endif
}
static inline void logTrace(const char *name)
{
#if defined(DEBUG)
logger.trace(name);
#endif
}
static inline void logDebug(const char *name, const char *format, ...)
{
#if defined(DEBUG)
va_list arguments;
va_start(arguments, format);
logger.debug(name, format, arguments);
va_end(arguments);
#endif
}
static inline void logDump(const char *name, const char *data, int size)
{
#if defined(TEST)
logger.dump(name, data, size);
#endif
}
#endif /* Logger_H */
############################################################################
# #
# Copyright (c) 2001, 2005 NoMachine, http://www.nomachine.com. #
# #
# NXCOMP, NX protocol compression and NX extensions to this software #
# are copyright of NoMachine. Redistribution and use of the present #
# software is allowed according to terms specified in the file LICENSE #
# which comes in the source distribution. #
# #
# Check http://www.nomachine.com/licensing.html for applicability. #
# #
# NX and NoMachine are trademarks of Medialogic S.p.A. #
# #
# All rights reserved. #
# #
############################################################################
#
# Get values from configure script.
#
VERSION=@VERSION@
LIBVERSION=@LIBVERSION@
#
# We would really like to enable all warnings, -Wredundant-decls,
# though, gives a warning caused by pthread.h and unistd.h and
# GCC 3.4 was changed in a way that it now complains about some
# of the -W directives we used before (-Wmissing-declarations,
# -Wnested-externs, -Wstrict-prototypes and -Wmissing-prototypes).
#
CXX = @CXX@
CXXFLAGS = @CXXFLAGS@ @X_CFLAGS@ @DEFS@ \
-Wall -Wpointer-arith
CXXINCLUDES =
CXXDEFINES =
#
# C programs don't share the C++ flags. They should
# have their own @CCFLAGS@.
#
CC = @CC@
CCFLAGS = @X_CFLAGS@ @DEFS@ \
-Wall -Wpointer-arith
CCINCLUDES =
CCDEFINES =
LDFLAGS = @LDFLAGS@
LIBS = @LIBS@ -L/usr/X11R6/lib -lX11
#
# Only if THREADS is defined.
#
# LIBS = $(LIBS) -lpthread
#
#
# Only if you want ElectricFence.
#
# LIBS = $(LIBS) -lefence
#
#
# Only if you want mpatrol.
#
# LIBS = $(LIBS) -lmpatrol -lbfd -liberty
#
srcdir = @srcdir@
prefix = @prefix@
exec_prefix = @exec_prefix@
bindir = @bindir@
man1dir = @mandir@/man1
VPATH = @srcdir@
INSTALL = @INSTALL@
INSTALL_PROGRAM = @INSTALL_PROGRAM@
INSTALL_DATA = @INSTALL_DATA@
#
# This should be autodetected.
#
MAKEDEPEND = @MAKEDEPEND@
DEPENDINCLUDES = -I/usr/include/g++ -I/usr/include/g++-3
.SUFFIXES: .cpp.c
.cpp.o:
$(CXX) -c $(CXXFLAGS) $(CXXINCLUDES) $(CXXDEFINES) $<
.c.o:
$(CC) -c $(CCFLAGS) $(CCINCLUDES) $(CCDEFINES) $<
LIBRARY = Xcompshad
LIBNAME = lib$(LIBRARY)
LIBFULL = lib$(LIBRARY).so.$(VERSION)
LIBLOAD = lib$(LIBRARY).so.$(LIBVERSION)
LIBSHARED = lib$(LIBRARY).so
LIBARCHIVE = lib$(LIBRARY).a
MSRC = Main.c
CSRC =
CXXSRC = Core.cpp \
Input.cpp \
Logger.cpp \
Shadow.cpp \
X11.cpp \
Win.cpp \
Updater.cpp \
Manager.cpp
MOBJ = $(MSRC:.c.cpp=.o)
COBJ = $(CSRC:.c=.o)
CXXOBJ = $(CXXSRC:.cpp=.o)
MLIBS = -L. -lXShadow
ifeq ($(findstring -lgdi32,$(LIBS)),-lgdi32)
# We need a smarter way to detect windows
# platform.
LIBDLL = cyg$(LIBRARY).dll
LIBDLLSTATIC = lib${LIBRARY}.dll.a
all: depend $(LIBARCHIVE) $(LIBDLL)
else
EXTRALIBS = -lXtst -lXrandr -lXdamage
all: depend $(LIBFULL) $(LIBLOAD) $(LIBSHARED) $(LIBARCHIVE)
endif
$(LIBFULL): $(CXXOBJ) $(COBJ)
$(CXX) -o $@ $(LDFLAGS) $(CXXOBJ) $(COBJ) $(LIBS) $(EXTRALIBS)
$(LIBLOAD): $(LIBFULL)
rm -f $(LIBLOAD)
ln -s $(LIBFULL) $(LIBLOAD)
$(LIBSHARED): $(LIBFULL)
rm -f $(LIBSHARED)
ln -s $(LIBFULL) $(LIBSHARED)
$(LIBARCHIVE): $(CXXOBJ) $(COBJ)
rm -f $(LIBARCHIVE)
ar clq $(LIBARCHIVE) $(CXXOBJ) $(COBJ)
ranlib $(LIBARCHIVE)
$(LIBDLL): $(LIBARCHIVE)
$(CC) -o $@ \
-shared \
-Wl,--out-implib=$(LIBDLLSTATIC) \
-Wl,--export-all-symbols \
-Wl,--enable-auto-import \
-Wl,--whole-archive ${LIBARCHIVE} \
-Wl,--no-whole-archive \
${LIBS} -L/usr/X11R6/lib
$(PROGRAM): $(MOBJ) $(COBJ) $(CXXOBJ) $(LIBDLL)
# $(CC) $(CCFLAGS) -o $@ $(MOBJ) $(MLIBS)
depends: depend.status
depend: depend.status
depend.status:
if [ -x $(MAKEDEPEND) ] ; then \
$(MAKEDEPEND) $(CXXINCLUDES) $(CCINCLUDES) \
$(DEPENDINCLUDES) -f Makefile $(MSRC) $(CSRC) $(CXXSRC) 2>/dev/null; \
fi
touch depend.status
install: install.bin install.man
install.bin:
install.man:
clean:
-rm -f *~ *.o *.bak st?????? core core.* *.out.* *.exe.stackdump \
$(LIBFULL) $(LIBLOAD) $(LIBSHARED) $(LIBARCHIVE) $(LIBDLL) $(LIBDLLSTATIC) $(PROGRAM) $(PROGRAM).exe
distclean: clean
-rm -rf config.status config.log config.cache depend.status Makefile tags autom4te.cache
/**************************************************************************/
/* */
/* Copyright (c) 2001, 2007 NoMachine, http://www.nomachine.com/. */
/* */
/* NXCOMPSHAD, NX protocol compression and NX extensions to this software */
/* are copyright of NoMachine. Redistribution and use of the present */
/* software is allowed according to terms specified in the file LICENSE */
/* which comes in the source distribution. */
/* */
/* Check http://www.nomachine.com/licensing.html for applicability. */
/* */
/* NX and NoMachine are trademarks of Medialogic S.p.A. */
/* */
/* All rights reserved. */
/* */
/**************************************************************************/
#include <X11/Xlib.h>
#include <X11/Xutil.h>
#include <X11/keysym.h>
#include <string.h>
#define PANIC
#define WARNING
#undef TEST
#undef DEBUG
#include "Manager.h"
#include "Logger.h"
UpdateManager::UpdateManager(int w, int h, char *f, Input *i)
: width_(w), height_(h), frameBuffer_(f), input_(i)
{
logTrace("UpdateManager::UpdateManager");
nUpdater = 0;
updaterVector = NULL;
updateManagerRegion_ = NULL;
}
UpdateManager::~UpdateManager()
{
logTrace("UpdateManager::~UpdateManager");
for (int i = 0; i < nUpdater; i++)
{
delete updaterVector[i];
}
delete [] updaterVector;
}
Updater *UpdateManager::createUpdater(char *displayName, Display *display)
{
Updater *updater = new Updater(displayName, display);
if (updater == NULL)
{
logError("UpdateManager::createUpdater", ESET(ENOMEM));
return NULL;
}
if (updater -> init(width_, height_, frameBuffer_, input_) == -1)
{
logError("UpdateManager::createUpdater", EGET());
delete updater;
return NULL;
}
return updater;
}
UpdaterHandle UpdateManager::addUpdater(char *displayName, Display *display)
{
Updater *newUpdater = createUpdater(displayName, display);
if (newUpdater == NULL)
{
logError("UpdateManager::addUpdater", EGET());
return NULL;
}
Updater **newUpdaterVector = new Updater*[nUpdater + 1];
if (newUpdaterVector == NULL)
{
logError("UpdateManager::addUpdater", ESET(ENOMEM));
delete newUpdater;
return NULL;
}
for (int i = 0; i < nUpdater; i++)
{
newUpdaterVector[i] = updaterVector[i];
}
newUpdaterVector[nUpdater] = newUpdater;
delete [] updaterVector;
updaterVector = newUpdaterVector;
nUpdater++;
logTest("UpdateManager::AddUpdater", "Number of updaters [%d].", nUpdater);
return reinterpret_cast<UpdaterHandle>(newUpdater);
}
int UpdateManager::removeAllUpdaters()
{
logTest("UpdateManager::removeAllUpdaters", "Number of updaters [%d].", nUpdater);
int nullUpdaters = 0;
for (int i = nUpdater; i > 0; i--)
{
if (removeUpdater(reinterpret_cast<UpdaterHandle>(updaterVector[i - 1])) == 0)
{
nullUpdaters++;
}
}
if (nUpdater == 0)
{
return 1;
}
if (nUpdater == nullUpdaters)
{
logTest("UpdateManager::removeAllUpdaters", "Ignored null records in Updater vector.");
return 0;
}
logTest("UpdateManager::removeAllUpdaters", "Failed to remove some updaters.");
return -1;
}
int UpdateManager::removeUpdater(UpdaterHandle handle)
{
Updater * const updater = (Updater*) handle;
logTest("UpdateManager::removeUpdater", "Removing Updater [%p].", updater);
if (updater == NULL)
{
return 0;
}
for (int i = 0; i < nUpdater; i++)
{
if (updater == updaterVector[i])
{
updaterVector[i] = updaterVector[nUpdater - 1];
nUpdater--;
delete updater;
return 1;
}
}
logTest("UpdateManager::removeUpdater", "Couldn't find Updater [%p].", updater);
return -1;
}
void UpdateManager::addRegion(Region region)
{
logTrace("UpdateManager::addRegion");
for (int i = 0; i < nUpdater; i++)
{
updaterVector[i] -> addRegion(region);
}
XDestroyRegion(region);
}
void UpdateManager::update()
{
logTrace("UpdateManager::update");
for (int i = 0; i < nUpdater; i++)
{
/*updaterVector[i] -> update();*/
if (updaterVector[i] -> getUpdateRegion())
{
logDebug("UpdateManager::update", "pRegion [%p] rect[%ld].",
updaterVector[i] -> getUpdateRegion(), (updaterVector[i] -> getUpdateRegion()) -> numRects);
updateManagerRegion_ = updaterVector[i] -> getUpdateRegion();
//
// FIXME: Remove me.
//
for (int j = 0; j < updateManagerRegion_ -> numRects; j++)
{
int x = updateManagerRegion_ -> rects[j].x1;
int y = updateManagerRegion_ -> rects[j].y1;
unsigned int width = updateManagerRegion_ -> rects[j].x2 - updateManagerRegion_ -> rects[j].x1;
unsigned int height = updateManagerRegion_ -> rects[j].y2 - updateManagerRegion_ -> rects[j].y1;
logDebug("UpdateManager::update", "x[%d]y[%d]width[%u]height[%u], updateManagerRegion_[%p]",
x, y, width, height, updateManagerRegion_);
}
}
}
}
void UpdateManager::handleInput()
{
logTrace("UpdateManager::handleInput");
for (int i = 0; i < nUpdater; i++)
{
try
{
updaterVector[i] -> handleInput();
}
catch (UpdaterClosing u)
{
logTest("UpdateManager::handleInput", "Catched exception UpdaterClosing().");
removeUpdater((UpdaterHandle)updaterVector[i]);
//
// Now the i-element of the updaterVector
// is changed. We don't want to skip it.
//
i--;
}
}
}
void UpdateManager::newRegion()
{
logTrace("UpdateManager::newRegion");
for (int i = 0; i < nUpdater; i++)
{
updaterVector[i] -> newRegion();
}
}
/**************************************************************************/
/* */
/* Copyright (c) 2001, 2007 NoMachine, http://www.nomachine.com/. */
/* */
/* NXCOMPSHAD, NX protocol compression and NX extensions to this software */
/* are copyright of NoMachine. Redistribution and use of the present */
/* software is allowed according to terms specified in the file LICENSE */
/* which comes in the source distribution. */
/* */
/* Check http://www.nomachine.com/licensing.html for applicability. */
/* */
/* NX and NoMachine are trademarks of Medialogic S.p.A. */
/* */
/* All rights reserved. */
/* */
/**************************************************************************/
#ifndef UpdateManager_H
#define UpdateManager_H
#include <X11/Xlib.h>
#include "Updater.h"
#include "Regions.h"
#include "Input.h"
typedef char* UpdaterHandle;
class UpdateManager
{
public:
UpdateManager(int, int, char *, Input *);
~UpdateManager();
void handleInput();
void addRegion(Region);
void update();
UpdaterHandle addUpdater(char *displayName, Display *display);
int removeUpdater(UpdaterHandle);
int removeAllUpdaters();
int numberOfUpdaters();
int getWidth();
int getHeight();
char *getBuffer();
Region getUpdateManagerRegion();
void destroyUpdateManagerRegion();
void newRegion();
private:
Updater *createUpdater(char *displayName, Display *display);
int width_;
int height_;
char *frameBuffer_;
Input *input_;
int nUpdater;
Updater **updaterVector;
Region updateManagerRegion_;
};
inline int UpdateManager::numberOfUpdaters()
{
return nUpdater;
}
inline int UpdateManager::getWidth()
{
return width_;
}
inline int UpdateManager::getHeight()
{
return height_;
}
inline char *UpdateManager::getBuffer()
{
return frameBuffer_;
}
inline Region UpdateManager::getUpdateManagerRegion()
{
return updateManagerRegion_;
}
inline void UpdateManager::destroyUpdateManagerRegion()
{
if (updateManagerRegion_ != NULL)
{
XDestroyRegion(updateManagerRegion_);
updateManagerRegion_ = NULL;
}
}
#endif /* UpdateManager_H */
/**************************************************************************/
/* */
/* Copyright (c) 2001, 2007 NoMachine, http://www.nomachine.com/. */
/* */
/* NXCOMPSHAD, NX protocol compression and NX extensions to this software */
/* are copyright of NoMachine. Redistribution and use of the present */
/* software is allowed according to terms specified in the file LICENSE */
/* which comes in the source distribution. */
/* */
/* Check http://www.nomachine.com/licensing.html for applicability. */
/* */
/* NX and NoMachine are trademarks of Medialogic S.p.A. */
/* */
/* All rights reserved. */
/* */
/**************************************************************************/
#ifndef Misc_H
#define Misc_H
#include <iostream.h>
#include <errno.h>
#include <string.h>
//
// Error handling macros.
//
#define ESET(e) (errno = (e))
#define EGET() (errno)
#define ESTR() strerror(errno)
//
// Log file.
//
extern ostream *logofs;
#endif /* Misc_H */
/**************************************************************************/
/* */
/* Copyright (c) 2001, 2007 NoMachine, http://www.nomachine.com/. */
/* */
/* NXCOMPSHAD, NX protocol compression and NX extensions to this software */
/* are copyright of NoMachine. Redistribution and use of the present */
/* software is allowed according to terms specified in the file LICENSE */
/* which comes in the source distribution. */
/* */
/* Check http://www.nomachine.com/licensing.html for applicability. */
/* */
/* NX and NoMachine are trademarks of Medialogic S.p.A. */
/* */
/* All rights reserved. */
/* */
/**************************************************************************/
#ifndef Poller_H
#define Poller_H
#if defined(__CYGWIN32__) || defined(WIN32)
#include "Win.h"
#else
#include "X11.h"
#endif
#endif /* Poller_H */
/**************************************************************************/
/* */
/* Copyright (c) 2001, 2007 NoMachine, http://www.nomachine.com/. */
/* */
/* NXCOMPSHAD, NX protocol compression and NX extensions to this software */
/* are copyright of NoMachine. Redistribution and use of the present */
/* software is allowed according to terms specified in the file LICENSE */
/* which comes in the source distribution. */
/* */
/* Check http://www.nomachine.com/licensing.html for applicability. */
/* */
/* NX and NoMachine are trademarks of Medialogic S.p.A. */
/* */
/* All rights reserved. */
/* */
/**************************************************************************/
#ifndef Region_H
#define Region_H
#include <X11/Xlib.h>
#include <X11/Xutil.h>
typedef struct {
short x1, x2, y1, y2;
} Box, BOX, BoxRec, *BoxPtr;
typedef struct _XRegion {
long size;
long numRects;
BOX *rects;
BOX extents;
};
#endif /* Region_H */
/**************************************************************************/
/* */
/* Copyright (c) 2001, 2007 NoMachine, http://www.nomachine.com. */
/* */
/* NXCOMPSHAD, NX protocol compression and NX extensions to this software */
/* are copyright of NoMachine. Redistribution and use of the present */
/* software is allowed according to terms specified in the file LICENSE */
/* which comes in the source distribution. */
/* */
/* Check http://www.nomachine.com/licensing.html for applicability. */
/* */
/* NX and NoMachine are trademarks of Medialogic S.p.A. */
/* */
/* All rights reserved. */
/* */
/**************************************************************************/
#include <signal.h>
#include <string.h>
#define PANIC
#define WARNING
#undef TEST
#undef DEBUG
#include "Logger.h"
#include "Shadow.h"
#include "Poller.h"
#include "Manager.h"
ShadowOptions NXShadowOptions = {1, 1, -1};
static int mirrorException = 0;
static UpdateManager *updateManager;
static Poller *poller;
static Input *input;
int NXShadowRemoveAllUpdaters();
inline bool NXShadowNotInitialized()
{
//
// updateManager depends on input and poller.
// So this test seem redundant.
//
// return (input == NULL) || (poller == NULL) || (updateManager == NULL);
//
return (updateManager == NULL);
}
#ifdef NEED_SIGNAL_HANDLER
static void NXSignalHandler(int signal)
{
logTest("NXSignalHandler", "Got signal [%d]", signal);
if (signal == SIGINT)
{
mirrorException = 1;
}
else if (signal == SIGTERM)
{
mirrorException = 1;
}
}
static int NXInitSignal()
{
logTrace("NXInitSignal");
struct sigaction sa;
sa.sa_handler = NXSignalHandler;
sigfillset(&sa.sa_mask);
sa.sa_flags = 0;
int res;
while ((res = sigaction(SIGINT, &sa, NULL)) == -1 &&
errno == EINTR);
if (res == -1)
{
logError("NXInitSignal", EGET());
return -1;
}
return 1;
}
#endif
static void NXHandleException()
{
if (mirrorException)
{
mirrorException = 0;
NXShadowRemoveAllUpdaters();
}
}
static int NXCreateInput(char *keymap, char *shadowDisplayName)
{
logTrace("NXCreateInput");
input = new Input;
if (input == NULL)
{
logError("NXCreateInput", ESET(ENOMEM));
return -1;
}
input -> setKeymap(keymap);
input -> setShadowDisplayName(shadowDisplayName);
return 1;
}
static int NXCreatePoller(Display *display, Display **shadowDisplay)
{
logTrace("NXCreatePoller");
if (input == NULL)
{
logError("NXCreatePoller", ESET(EBADFD));
return -1;
}
poller = new Poller(input,display);
if (poller == NULL)
{
logError("NXCreatePoller", ESET(ENOMEM));
return -1;
}
if (poller -> init() == -1)
{
logTest("NXCreatePoller", "Failed to initialize poller.");
return -1;
}
*shadowDisplay = poller -> getShadowDisplay();
logTest("NXCreatePoller", "Poller geometry [%d, %d], ShadowDisplay[%p].", poller -> width(),
poller -> height(), (Display *) *shadowDisplay);
return 1;
}
static int NXCreateUpdateManager()
{
logTrace("NXCreateUpdateManager");
if (input == NULL || poller == NULL)
{
logError("NXCreateUpdateManager", ESET(EBADFD));
return -1;
}
updateManager = new UpdateManager(poller -> width(), poller -> height(),
poller -> getFrameBuffer(), input);
if (updateManager == NULL)
{
logError("NXCreateUpdateManager", ESET(ENOMEM));
return -1;
}
return 1;
}
void NXShadowResetOptions()
{
NXShadowOptions.optionShmExtension = 1;
NXShadowOptions.optionDamageExtension = 1;
}
//
// Exported functions.
//
int NXShadowHasUpdaters()
{
logTrace("NXShadowHasUpdaters");
return (updateManager && updateManager -> numberOfUpdaters()) ? 1 : 0;
}
int NXShadowRemoveAllUpdaters()
{
logTrace("NXShadowRemoveAllUpdaters");
return updateManager ? updateManager -> removeAllUpdaters() : 0;
}
int NXShadowRemoveUpdater(UpdaterHandle handle)
{
logTrace("NXShadowRemoveUpdater");
return updateManager ? updateManager -> removeUpdater(handle) : 0;
}
UpdaterHandle NXShadowAddUpdater(char *displayName)
{
logTrace("NXShadowAddUpdater");
return updateManager ? updateManager -> addUpdater(displayName, NULL) : NULL;
}
int NXShadowAddUpdaterDisplay(void *dpy, int *w, int *h, unsigned char *d)
{
Display *display = reinterpret_cast<Display*>(dpy);
logTrace("NXShadowAddUpdaterDisplay");
if ((updateManager ? updateManager -> addUpdater(NULL, display) : NULL) == NULL)
{
logTest("NXShadowAddUpdaterDisplay", "Error");
return 0;
}
*w = updateManager -> getWidth();
*h = updateManager -> getHeight();
*d = poller -> depth();
return 1;
}
int NXShadowCreate(void *dpy, char *keymap, char* shadowDisplayName, void **shadowDpy)
{
logTrace("NXShadowCreate");
Display *display = reinterpret_cast<Display*>(dpy);
Display **shadowDisplay = reinterpret_cast<Display**>(shadowDpy);
/* if (NXInitSignal() != 1)
{
logError("NXShadowCreate", EGET());
return -1;
}*/
if (NXCreateInput(keymap, shadowDisplayName) != 1)
{
logError("NXShadowCreate", EGET());
return -1;
}
if (NXCreatePoller(display, shadowDisplay) != 1)
{
logTest("NXShadowCreate", "NXCreatePoller failed.");
return -1;
}
if (NXCreateUpdateManager() != 1)
{
logError("NXShadowCreate", EGET());
return -1;
}
return 1;
}
#if !defined(__CYGWIN32__) && !defined(WIN32)
void NXShadowSetDisplayUid(int uid)
{
NXShadowOptions.optionShadowDisplayUid = uid;
}
void NXShadowDisableShm(void)
{
logUser("NXShadowDisableShm: Disabling SHM.\n");
NXShadowOptions.optionShmExtension = 0;
}
void NXShadowDisableDamage(void)
{
NXShadowOptions.optionDamageExtension = 0;
}
#endif
void NXShadowDestroy()
{
if (poller)
{
delete poller;
poller = NULL;
}
if (updateManager)
{
delete updateManager;
updateManager = NULL;
}
if (input)
{
delete input;
input = NULL;
}
}
void NXShadowHandleInput()
{
logTrace("NXShadowHandleInput");
if (NXShadowNotInitialized())
{
logError("NXShadowHandleInput - NXShadow not properly initialized.", ESET(EBADFD));
return;
}
NXHandleException();
updateManager -> handleInput();
poller -> handleInput();
}
int NXShadowHasChanged(int (*callback)(void *), void *arg, int *suspended)
{
int result;
logTrace("NXShadowHasChanged");
if (NXShadowNotInitialized())
{
logError("NXShadowHasChanged - NXShadow not properly initialized.", ESET(EBADFD));
return -1;
}
//
// FIXME
//updateManager -> destroyUpdateManagerRegion();
//
updateManager -> newRegion();
#if !defined(__CYGWIN32__) && !defined(WIN32)
poller -> getEvents();
#endif
result = poller -> isChanged(callback, arg, suspended);
if (result == 1)
{
updateManager -> addRegion(poller -> lastUpdatedRegion());
return 1;
}
else if (result == -1)
{
logTest("NXShadowHasChanged", "Scanline error.");
return -1;
}
return 0;
}
void NXShadowExportChanges(long *numRects, char **pBox)
{
Region pReg;
logTrace("NXShadowExportChanges");
if (NXShadowNotInitialized())
{
logError("NXShadowExportChanges - NXShadow not properly initialized.", ESET(EBADFD));
}
updateManager -> update();
pReg = updateManager -> getUpdateManagerRegion();
*numRects = pReg -> numRects;
*pBox = (char *)pReg -> rects;
logTest("NXShadowExportChanges", "numRects [%ld] pBox[%p], pReg->numRects[%ld], rects[%p], size[%lu]",
*numRects, *pBox, pReg -> numRects, &(pReg -> rects -> x2),
(unsigned long) sizeof(pReg -> rects -> x2));
}
void NXShadowEvent(Display *display, XEvent event)
{
poller -> handleEvent(display, &event);
}
#ifdef __CYGWIN32__
int NXShadowCaptureCursor(unsigned int wnd, void *vis)
{
Window window = (Window)wnd;
Visual *visual = reinterpret_cast<Visual*>(vis);
logTrace("NXShadowCaptureCursor");
logTest("NXShadowCaptureCursor","Init");
return poller -> updateCursor(window, visual);
}
#endif
void NXShadowUpdateBuffer(void **buffer)
{
char **fBuffer = reinterpret_cast<char **>(buffer);
if (*fBuffer != NULL)
{
poller -> destroyFrameBuffer();
poller -> init();
}
*fBuffer = poller -> getFrameBuffer();
logTest("NXShadowUpdateBuffer","New frame buffer [0x%p]", (void *)*fBuffer);
}
/**************************************************************************/
/* */
/* Copyright (c) 2001, 2007 NoMachine, http://www.nomachine.com. */
/* */
/* NXCOMPSHAD, NX protocol compression and NX extensions to this software */
/* are copyright of NoMachine. Redistribution and use of the present */
/* software is allowed according to terms specified in the file LICENSE */
/* which comes in the source distribution. */
/* */
/* Check http://www.nomachine.com/licensing.html for applicability. */
/* */
/* NX and NoMachine are trademarks of Medialogic S.p.A. */
/* */
/* All rights reserved. */
/* */
/**************************************************************************/
#ifndef Shadow_H
#define Shadow_H
#include <X11/Xlib.h>
#define NXShadowCorrectColor(length, buffer) \
\
{ \
unsigned short a; \
unsigned short b; \
unsigned short *shorts; \
int i; \
\
length >>= 1; \
shorts = (unsigned short *)buffer; \
for (i = 0; i < length ; i++) \
{ \
a = shorts[i]; \
\
b = a & 63; \
a <<= 1; \
a = (a & ~127) | b; \
\
shorts[i] = a; \
} \
}
#ifdef __cplusplus
extern "C" {
#endif
typedef char* UpdaterHandle;
typedef struct _ShadowOptions
{
char optionShmExtension;
char optionDamageExtension;
int optionShadowDisplayUid;
} ShadowOptions;
extern ShadowOptions NXShadowOptions;
extern int NXShadowCreate(void *, char *, char *, void **);
extern void NXShadowDestroy(void);
/*
* Use an already opened Display connection.
* We use <void *> instead of <Display *> to avoid
* useless dependences from Xlib headers.
*/
extern int NXShadowAddUpdaterDisplay(void *display, int *width, int *height,
unsigned char *depth);
extern UpdaterHandle NXShadowAddUpdater(char *displayName);
extern int NXShadowRemoveUpdater(UpdaterHandle handle);
extern int NXShadowRemoveAllUpdaters(void);
extern void NXShadowHandleInput(void);
extern int NXShadowHasChanged(int (*)(void *), void *, int *);
extern void NXShadowExportChanges(long *, char **);
extern int NXShadowHasUpdaters(void);
extern int NXShadowCaptureCursor(unsigned int wnd, void *vis);
extern void NXShadowColorCorrect(int, int, unsigned int, unsigned int, char *);
extern void NXShadowUpdateBuffer(void **);
extern void NXShadowEvent(Display *, XEvent);
extern void NXShadowSetDisplayUid(int uid);
extern void NXShadowDisableShm(void);
extern void NXShadowDisableDamage(void);
#ifdef __cplusplus
}
#endif
#endif /* Shadow_H */
/**************************************************************************/
/* */
/* Copyright (c) 2001, 2007 NoMachine, http://www.nomachine.com/. */
/* */
/* NXCOMPSHAD, NX protocol compression and NX extensions to this software */
/* are copyright of NoMachine. Redistribution and use of the present */
/* software is allowed according to terms specified in the file LICENSE */
/* which comes in the source distribution. */
/* */
/* Check http://www.nomachine.com/licensing.html for applicability. */
/* */
/* NX and NoMachine are trademarks of Medialogic S.p.A. */
/* */
/* All rights reserved. */
/* */
/**************************************************************************/
#include <X11/Xlib.h>
#include <X11/Xutil.h>
#include <X11/keysym.h>
#include <string.h>
#define PANIC
#define WARNING
#undef TEST
#undef DEBUG
//
#include <stdio.h>
//
#include "Updater.h"
#include "Logger.h"
Updater::Updater(char *displayName, Display *display)
{
logTrace("Updater::Updater");
displayName_ = displayName;
display_ = display;
closeDisplay_ = false;
image_ = NULL;
updateRegion_ = NULL;
buffer_ = NULL;
}
Updater::~Updater()
{
logTrace("Updater::~Updater");
if (input_)
{
int removedEvents = input_ -> removeAllEvents(display_);
logTest("Updater::~Updater", "Removed events in input queue is [%d].", removedEvents);
}
if (display_)
{
XDestroyWindow(display_, window_);
XFreePixmap(display_, pixmap_);
if (closeDisplay_)
{
XCloseDisplay(display_);
}
}
if (image_)
{
image_ -> data = NULL;
XDestroyImage(image_);
}
if (updateRegion_)
{
XDestroyRegion(updateRegion_);
}
}
int Updater::init(int width, int height, char *fb, Input *input)
{
logTrace("Updater::init");
if (fb == NULL || input == NULL || width <= 0 || height <= 0)
{
logError("Updater::init", ESET(EINVAL));
return -1;
}
width_ = width;
height_ = height;
buffer_ = fb;
input_ = input;
/*
if (display_ == NULL)
{
display_ = XOpenDisplay(displayName_);
closeDisplay_ = true;
if (display_ == NULL)
{
logError("Updater::init", ESET(ENOMSG));
return -1;
}
}
*/
depth_ = DefaultDepth(display_, DefaultScreen(display_));
if (depth_ == 8) bpl_ = width_;
else if (depth_ == 16) bpl_ = width_ * 2;
else bpl_ = width_ * 4;
logTest("Updater::init", "Server geometry [%d, %d] depth [%d] bpl [%d].", width_, height_, depth_, bpl_);
/* int bitmap_pad = 8;
image_ = XCreateImage(display_, DefaultVisual(display_, DefaultScreen(display_)), depth_, ZPixmap, 0,
buffer_, width_, height_, bitmap_pad, 0);
if (image_ == NULL)
{
logError("Updater::init", ESET(ENOMSG));
logTest("Updater::init", "Failed to create default image.");
return -1;
}
pixmap_ = XCreatePixmap(display_, DefaultRootWindow(display_), width_, height_, depth_);
unsigned int mask = CWBackPixmap | CWBorderPixel | CWEventMask;
XSetWindowAttributes attributes;
attributes.background_pixmap = pixmap_;
attributes.border_pixel = WhitePixel(display_, DefaultScreen(display_));
attributes.event_mask = KeyPressMask | KeyReleaseMask | ButtonPressMask | ButtonReleaseMask;
window_ = XCreateWindow(display_, DefaultRootWindow(display_),
0, 0, width_, height_, 0, depth_, InputOutput,
DefaultVisual(display_, DefaultScreen(display_)), mask, &attributes);
if (window_ == None)
{
logError("Updater::init", ESET(ENOMSG));
return -1;
}
XSizeHints *size_hints;
if ((size_hints = XAllocSizeHints()) == NULL)
{
logError("Updater::init", ESET(ENOMEM));
return -1;
}
size_hints -> flags = PMinSize | PMaxSize;
size_hints -> min_width = width_;
size_hints -> max_width = width_;
size_hints -> min_height = height_;
size_hints -> max_height = height_;
XSetWMNormalHints(display_, window_, size_hints);
XFree(size_hints);
Atom deleteWMatom = XInternAtom(display_, "WM_DELETE_WINDOW", 1);
XSetWMProtocols(display_, window_, &deleteWMatom, 1);
XMapWindow(display_, window_);*/
updateRegion_ = XCreateRegion();
logTest("Updater::init", "updateRegion_[%p]", updateRegion_);
return 1;
}
void Updater::addRegion(Region region)
{
//
// FIXME: Is this too paranoid ?
//
if (updateRegion_ == NULL)
{
logError("Updater::addRegion", ESET(EINVAL));
return;
}
XUnionRegion(region, updateRegion_, updateRegion_);
}
void Updater::update()
{
logTrace("Updater::update");
if (updateRegion_ == NULL)
{
logError("Updater::update", ESET(EINVAL));
return;
}
logTest("Updater::update", "Number of rectangles [%ld].", updateRegion_ -> numRects);
/* for (; updateRegion_ -> numRects > 0; updateRegion_ -> numRects--)
{
int n = updateRegion_ -> numRects - 1;
int x = updateRegion_ -> rects[n].x1;
int y = updateRegion_ -> rects[n].y1;
unsigned int width = updateRegion_ -> rects[n].x2 - updateRegion_ -> rects[n].x1;
unsigned int height = updateRegion_ -> rects[n].y2 - updateRegion_ -> rects[n].y1;
logDebug("Updater::update", "Sending rectangle: [%d, %d, %d, %d].", x, y, width, height);
//
// We need to update the extents.
//
int bitmap_pad;
if (depth_ == 32 || depth_ == 24)
{
bitmap_pad = 32;
}
else if (depth_ == 16)
{
if ((width & 1) == 0)
{
bitmap_pad = 32;
}
else
{
bitmap_pad = 16;
}
}
else if ((width & 3) == 0)
{
bitmap_pad = 32;
}
else if ((width & 1) == 0)
{
bitmap_pad = 16;
}
else
{
bitmap_pad = 8;
}*/
/* image_ -> bitmap_pad = bitmap_pad;*/
/* NXShadowCorrectColor(x, y, width, height);*/
/* XPutImage(display_, pixmap_, DefaultGC(display_, DefaultScreen(display_)),
image_, x, y, x, y, width, height);
XClearArea(display_, window_, x, y, width, height, 0);
}*/
//
// Should we reduces the box vector ?
//
// BOX *box = Xrealloc(updateRegion_ -> rects,
// updateRegion_ -> numRects == 0 ? sizeof(BOX) :
// updateRegion_ -> numRects * sizeof(BOX));
//
// if (box)
// {
// updateRegion_ -> rects = box;
// updateRegion_ -> size = 1;
// }
//
if (updateRegion_ -> numRects == 0)
{
updateRegion_ -> extents.x1 = 0;
updateRegion_ -> extents.y1 = 0;
updateRegion_ -> extents.x2 = 0;
updateRegion_ -> extents.y2 = 0;
}
else
{
//
// FIXME: We have to update the region extents.
//
logTest("Updater::update", "Region extents has not been updated.");
}
}
void Updater::handleInput()
{
logTrace("Updater::handleInput");
XEvent *event = new XEvent;
if (event == NULL)
{
logError("Updater::handleInput", ESET(ENOMEM));
return;
}
while (XCheckIfEvent(display_, event, anyEventPredicate, NULL))
{
switch (event -> type)
{
/* case ClientMessage:
{
Atom wmProtocols = XInternAtom(display_, "WM_PROTOCOLS", 0);
Atom wmDeleteWindow = XInternAtom(display_, "WM_DELETE_WINDOW", 0);
if (event -> xclient.message_type == wmProtocols &&
(Atom)event -> xclient.data.l[0] == wmDeleteWindow)
{
logTest("Updater::handleInput", "Got client message of type WM_PROTOCOLS and value WM_DELETE_WINDOW,"
" throwing exception UpdaterClosing.");
delete event;
throw UpdaterClosing();
}
else
{
logTest("Updater::handleInput", "Unexpected client message type [%ld] format [%d] first value [%ld]",
event -> xclient.message_type, event -> xclient.format, event -> xclient.data.l[0]);
}
break;
}*/
case KeyPress:
case KeyRelease:
case ButtonPress:
case ButtonRelease:
case MotionNotify:
{
input_ -> pushEvent(display_, event);
event = new XEvent;
if (event == NULL)
{
logError("Updater::handleInput", ESET(ENOMEM));
return;
}
break;
}
default:
{
logTest("Updater::handleInput", "Handling unexpected event [%d].", event -> type);
break;
}
}
}
delete event;
}
void Updater::newRegion()
{
if (updateRegion_ != NULL)
{
XDestroyRegion(updateRegion_);
}
updateRegion_ = XCreateRegion();
logTest("Updater::newRegion", "updateRegion_ [%p].", updateRegion_);
}
//
// Private functions.
//
/**************************************************************************/
/* */
/* Copyright (c) 2001, 2007 NoMachine, http://www.nomachine.com/. */
/* */
/* NXCOMPSHAD, NX protocol compression and NX extensions to this software */
/* are copyright of NoMachine. Redistribution and use of the present */
/* software is allowed according to terms specified in the file LICENSE */
/* which comes in the source distribution. */
/* */
/* Check http://www.nomachine.com/licensing.html for applicability. */
/* */
/* NX and NoMachine are trademarks of Medialogic S.p.A. */
/* */
/* All rights reserved. */
/* */
/**************************************************************************/
#ifndef Updater_H
#define Updater_H
#include <X11/Xlib.h>
#include "Regions.h"
#include "Input.h"
class UpdaterClosing {};
class Updater
{
public:
Updater(char *displayName, Display *display);
~Updater();
int init(int, int, char *, Input*);
void addRegion(Region r);
void update();
void handleInput();
XImage *getImage();
Region getUpdateRegion();
void newRegion();
private:
Input *input_;
static inline Bool anyEventPredicate(Display*, XEvent*, XPointer);
void handleKeyboardEvent(XEvent &event);
char *displayName_;
char *buffer_;
bool closeDisplay_;
Display *display_;
int depth_;
int width_;
int height_;
int bpl_;
Window window_;
XImage *image_;
Pixmap pixmap_;
Region updateRegion_;
};
Bool Updater::anyEventPredicate(Display*, XEvent*, XPointer)
{
return true;
}
inline XImage* Updater::getImage()
{
return image_;
}
inline Region Updater::getUpdateRegion()
{
return updateRegion_;
}
#endif /* Updater_H */
/**************************************************************************/
/* */
/* Copyright (c) 2001, 2007 NoMachine, http://www.nomachine.com/. */
/* */
/* NXCOMPSHAD, NX protocol compression and NX extensions to this software */
/* are copyright of NoMachine. Redistribution and use of the present */
/* software is allowed according to terms specified in the file LICENSE */
/* which comes in the source distribution. */
/* */
/* Check http://www.nomachine.com/licensing.html for applicability. */
/* */
/* NX and NoMachine are trademarks of Medialogic S.p.A. */
/* */
/* All rights reserved. */
/* */
/**************************************************************************/
#if defined(__CYGWIN32__) || defined(WIN32)
#include <X11/keysym.h>
#define PANIC
#define WARNING
#undef TEST
#undef DEBUG
#include "Poller.h"
#include "Logger.h"
Poller::Poller(Input *input, Display *display, int depth) : CorePoller(input, display)
{
logTrace("Poller::Poller");
screenDC_ = NULL;
screenBmp_ = NULL;
memoryDC_ = NULL;
pDIBbits_ = NULL;
DIBBuffer_ = NULL;
pKey_ = NULL;
pMouse_ = NULL;
path_ = NULL;
keymapName_ = input -> getKeymap();
keymap_ = NULL;
toggleButtonState_ = 0;
serverModifierState_ = 0;
display_ = display;
depth_ = DefaultDepth(display_, DefaultScreen(display_));
oldCursor_ = 0;
xCursor_ = 0;
}
Poller::~Poller()
{
logTrace("Poller::~Poller");
if (screenDC_)
{
BOOL result = ReleaseDC(NULL, screenDC_);
logTest("Poller::~Poller", "ReleaseDC returned [%d].", result);
screenDC_ = NULL;
}
if (memoryDC_)
{
BOOL result = DeleteDC(memoryDC_);
logTest("Poller::~Poller", "DeleteDC returned [%d].", result);
memoryDC_ = NULL;
}
if (screenBmp_)
{
BOOL result = DeleteObject(screenBmp_);
logTest("Poller::~Poller", "DeleteObject returned [%d].", result);
screenBmp_ = NULL;
}
if (DIBBuffer_)
{
logDebug("Poller::~Poller", "Delete DIBBuffer_ [%p].", DIBBuffer_);
delete [] DIBBuffer_;
}
if (pKey_)
{
logDebug("Poller::~Poller", " pKey_[%p].", pKey_);
delete [] pKey_;
}
if (pMouse_)
{
logDebug("Poller::~Poller", " pMouse_[%p].", pMouse_);
delete [] pMouse_;
}
if (keymap_)
{
logDebug("Poller::~Poller", " keymap_[%p].", keymap_);
delete [] keymap_;
}
}
int Poller::init()
{
logTrace("Poller::init");
int maxLengthArrayINPUT = 6;
platformOS();
pKey_ = new INPUT [maxLengthArrayINPUT];
if (pKey_ == NULL)
{
logError("Poller::init", ESET(ENOMEM));
return -1;
}
for (int i = 0; i < maxLengthArrayINPUT; i++)
{
pKey_[i].type = INPUT_KEYBOARD;
pKey_[i].ki.wVk = (WORD) 0;
pKey_[i].ki.time = (DWORD) 0;
pKey_[i].ki.dwExtraInfo = (DWORD) 0;
}
pMouse_ = new INPUT;
if (pMouse_ == NULL)
{
logError("Poller::init", ESET(ENOMEM));
return -1;
}
pMouse_ -> type = INPUT_MOUSE;
pMouse_ -> mi.dx = 0;
pMouse_ -> mi.dy = 0;
pMouse_ -> mi.mouseData = (DWORD) 0;
pMouse_ -> mi.time = 0;
pMouse_ -> mi.dwExtraInfo = (ULONG_PTR) NULL;
screenDC_ = GetDC(NULL);
if (screenDC_ == NULL)
{
logError("Poller::init", ESET(ENOMSG));
return -1;
}
switch(depth_)
{
case 8:
{
depth_ = 16;
break;
}
case 16:
{
depth_ = 16;
break;
}
case 24:
{
depth_ = 32;
break;
}
default:
{
logError("Poller::init", ESET(EINVAL));
return -1;
}
}
width_ = GetDeviceCaps(screenDC_, HORZRES);
height_ = GetDeviceCaps(screenDC_, VERTRES);
bpl_ = width_ * (depth_ >> 3);
bpp_ = (depth_ >> 3);
logTest("Poller::init", "Screen geometry is [%d, %d] depth is [%d] bpl [%d] bpp [%d].",
width_, height_, depth_, bpl_, bpp_);
logTest("Poller::init", "Got device context at [%p] screen size is (%d,%d).",
screenDC_, width_, height_);
memoryDC_ = CreateCompatibleDC(screenDC_);
if (memoryDC_ == NULL)
{
logError("Poller::init", ESET(ENOMSG));
return -1;
}
//
// Delete the old bitmap for the memory device.
//
HBITMAP bitmap = (HBITMAP) GetCurrentObject(memoryDC_, OBJ_BITMAP);
if (bitmap && DeleteObject(bitmap) == 0)
{
logError("Poller::init", ESET(ENOMSG));
}
//
// Bitmap header describing the bitmap we want to get from GetDIBits.
//
bmi_.bmiHeader.biSize = sizeof(BITMAPINFOHEADER);
bmi_.bmiHeader.biWidth = width_;
bmi_.bmiHeader.biHeight = -height_;
bmi_.bmiHeader.biPlanes = 1;
bmi_.bmiHeader.biBitCount = depth_;
bmi_.bmiHeader.biCompression = BI_RGB;
bmi_.bmiHeader.biSizeImage = 0;
bmi_.bmiHeader.biXPelsPerMeter = 0;
bmi_.bmiHeader.biYPelsPerMeter = 0;
bmi_.bmiHeader.biClrUsed = 0;
bmi_.bmiHeader.biClrImportant = 0;
screenBmp_ = CreateDIBSection(memoryDC_, &bmi_, DIB_RGB_COLORS, &pDIBbits_, NULL, 0);
ReleaseDC(NULL,memoryDC_);
if (screenBmp_ == NULL)
{
logTest ("Poller::init", "This video device is not supporting DIB section");
pDIBbits_ = NULL;
screenBmp_ = CreateCompatibleBitmap(screenDC_, width_, height_);
if (screenBmp_ == NULL)
{
logError("Poller::init", ESET(ENOMSG));
return -1;
}
if (SelectObject(memoryDC_, screenBmp_) == NULL)
{
logError("Poller::init", ESET(ENOMSG));
return -1;
}
}
else
{
logTest ("Poller::init", "Enabled the DIB section");
if (SelectObject(memoryDC_, screenBmp_) == NULL)
{
logError("Poller::init", ESET(ENOMSG));
return -1;
}
}
//
// Check if the screen device raster capabilities
// support the bitmap transfer.
//
if ((GetDeviceCaps(screenDC_, RASTERCAPS) & RC_BITBLT) == 0)
{
logTest("Poller::init", "This video device is not supporting the bitmap transfer.");
logError("Poller::init", ESET(ENOMSG));
return -1;
}
//
// Check if the memory device raster capabilities
// support the GetDIBits and SetDIBits functions.
//
if ((GetDeviceCaps(memoryDC_, RASTERCAPS) & RC_DI_BITMAP) == 0)
{
logTest("Poller::init", "This memory device is not supporting the GetDIBits and SetDIBits "
"function.");
logError("Poller::init", ESET(ENOMSG));
return -1;
}
if (GetDeviceCaps(screenDC_, PLANES) != 1)
{
logTest("Poller::init", "This video device has more than 1 color plane.");
logError("Poller::init", ESET(ENOMSG));
return -1;
}
return CorePoller::init();
}
//
// FIXME: Remove me.
//
void ErrorExit(LPTSTR lpszFunction)
{
LPVOID lpMsgBuf;
DWORD dw = GetLastError();
FormatMessage(
FORMAT_MESSAGE_ALLOCATE_BUFFER |
FORMAT_MESSAGE_FROM_SYSTEM,
NULL,
dw,
MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT),
(LPTSTR) &lpMsgBuf,
0, NULL );
logTest(lpszFunction, " Failed with error [%ld]: %s", dw, (char*)lpMsgBuf);
LocalFree(lpMsgBuf);
ExitProcess(dw);
}
//
// FIXME: End.
//
char *Poller::getRect(XRectangle r)
{
logTrace("Poller::getRect");
logDebug("Poller::getRect", "Going to retrive rectangle [%d, %d, %d, %d].",
r.x, r.y, r.width, r.height);
//
// The CAPTUREBLT operation could be a very
// cpu-consuming task. We should make some
// test to see how much it is expensive.
// Anyway we get tooltip windows and any
// other special effect not included with
// only the SRCCOPY operation.
//
if (BitBlt(memoryDC_, r.x, r.y, r.width, r.height,
screenDC_, r.x, r.y, SRCCOPY | CAPTUREBLT) == 0)
{
logError("Poller::getRect", ESET(ENOMSG));
logTest("Poller::getRect", "Failed to perform a bit-block transfer.");
logTest("Poller::getRect", "bit-block error=%lu", GetLastError());
return NULL;
}
// bmi_.bmiHeader.biWidth = r.width;
// bmi_.bmiHeader.biHeight = -r.height;
if (pDIBbits_ == NULL)
{
static long nPixel = 0;
if (nPixel < r.width * r.height)
{
if (DIBBuffer_)
{
delete [] DIBBuffer_;
}
nPixel = r.width * r.height;
DIBBuffer_ = new char [nPixel * bpp_];
if (DIBBuffer_ == NULL)
{
logError("Poller::getRect", ESET(ENOMEM));
nPixel = 0;
return NULL;
}
}
if (GetDIBits(memoryDC_, screenBmp_, height_ - r.height - r.y, r.height,
DIBBuffer_, &bmi_, DIB_RGB_COLORS) == 0)
{
logError("Poller::getRect", ESET(ENOMSG));
logTest("Poller::getRect", "Failed to retrieve the screen bitmap.");
return NULL;
}
return DIBBuffer_;
}
else
{
return (char *) pDIBbits_ + r.y * bpl_ + r.x * bpp_;
}
}
void Poller::handleKeyboardEvent(Display *display, XEvent *event)
{
KeySym keysym;
char *keyname = new char [31];
keyTranslation tr = {0, 0};
unsigned char scancode = 0;
int lengthArrayINPUT = 0;
if (XLookupString((XKeyEvent *) event, keyname, 30, &keysym, NULL) > 0)
{
logTest("Poller::handleKeyboardEvent", "keyname %s, keysym [%x]", keyname, (unsigned int)keysym);
}
if (specialKeys(keysym, event -> xkey.state, event -> type) == 1)
{
delete[] keyname;
return;
}
tr = xkeymapTranslateKey(keysym, event -> xkey.keycode, event -> xkey.state);
scancode = tr.scancode;
logTest("Poller::handleKeyboardEvent", "keyname [%s] scancode [0x%x], keycode[0x%x], keysym [%x]", keyname,
tr.scancode, event ->xkey.keycode, (unsigned int)keysym);
if (scancode == 0)
{
delete[] keyname;
return;
}
if (event -> type == KeyPress)
{
int test1 = MapVirtualKey(scancode, MAPVK_VSC_TO_VK_EX);
int test2 = MapVirtualKey(0x24, MAPVK_VSC_TO_VK_EX);
if (test1 == test2)
{
simulateCtrlAltDel();
}
if (isModifier(scancode) == 0)
{
savedServerModifierState_ = serverModifierState_;
}
ensureServerModifiers(tr, &lengthArrayINPUT);
if (sendInput(scancode, 1, &lengthArrayINPUT) == 0)
{
logTest("Poller::handleKeyboardEvent", "lengthArrayINPUT [%d].", lengthArrayINPUT);
}
restoreServerModifiers(scancode);
}
else if (event -> type == KeyRelease)
{
if (sendInput(scancode, 0, &lengthArrayINPUT) == 0)
{
logTest("Poller::handleKeyboardEvent", "lengthArrayINPUT [%d].", lengthArrayINPUT);
}
}
updateModifierState(scancode, (event -> type == KeyPress));
delete[] keyname;
}
void Poller::handleMouseEvent(Display *display, XEvent *event)
{
DWORD flg = 0;
DWORD whl = 0;
if (event -> type == ButtonPress)
{
logTest("Poller::handleMouseEvent", "ButtonPress.\n");
switch (event -> xbutton.button)
{
case Button1:
{
flg = MOUSEEVENTF_LEFTDOWN;
break;
}
case Button2:
{
flg = MOUSEEVENTF_MIDDLEDOWN;
break;
}
case Button3:
{
flg = MOUSEEVENTF_RIGHTDOWN;
break;
}
case Button4:
{
flg = MOUSEEVENTF_WHEEL;
whl = WHEEL_DELTA;
pMouse_ -> mi.mouseData = whl;
break;
}
case Button5:
{
flg = MOUSEEVENTF_WHEEL;
whl = (DWORD) (-WHEEL_DELTA);
pMouse_ -> mi.mouseData = whl;
break;
}
}
}
else if (event -> type == ButtonRelease)
{
switch (event -> xbutton.button)
{
case Button1:
{
flg = MOUSEEVENTF_LEFTUP;
break;
}
case Button2:
{
flg = MOUSEEVENTF_MIDDLEUP;
break;
}
case Button3:
{
flg = MOUSEEVENTF_RIGHTUP;
break;
}
case Button4:
{
flg = MOUSEEVENTF_WHEEL;
whl = 0;
pMouse_ -> mi.mouseData = whl;
break;
}
case Button5:
{
flg = MOUSEEVENTF_WHEEL;
whl = 0;
pMouse_ -> mi.mouseData = whl;
break;
}
}
}
else if (event -> type == MotionNotify)
{
logTest("Poller::handleMouseEvent", "SetCursor - MotionNotify");
SetCursorPos(event -> xmotion.x, event -> xmotion.y);
}
if (flg > 0)
{
// logTest("Poller::handleMouseEvent", "SetCursor - flg > 0");
//
// FIXME: move the cursor to the pace the event occurred
//
SetCursorPos(event -> xbutton.x, event -> xbutton.y);
//
// FIXME: Remove me: send the click/release event
// mouse_event(flg, 0, 0, whl, (ULONG_PTR)NULL);
//
pMouse_ -> mi.dwFlags = flg;
if (SendInput(1, pMouse_, sizeof(INPUT)) == 0)
{
logTest("Poller::handleMouseEvent", "Failed SendInput");
}
}
}
int Poller::updateCursor(Window wnd, Visual* vis)
{
BYTE *mBits, *andBits, *xorBits;
logTrace("Poller::Cursor");
//
// Retrieve mouse cursor handle.
//
CURSORINFO cursorInfo;
cursorInfo.cbSize = sizeof(CURSORINFO);
if (GetCursorInfo(&cursorInfo) == 0)
{
logTest("Poller::Cursor", "GetCursorInfo() failed [%u].\n", (unsigned int)GetLastError());
LocalFree(&cursorInfo);
return -1;
}
HCURSOR hCursor = cursorInfo.hCursor;
if (hCursor == 0)
{
logTest("Poller::Cursor","Cursor Handle is NULL. Error[%u].\n", (unsigned int)GetLastError());
return 1;
}
if (hCursor == oldCursor_)
{
LocalFree(&cursorInfo);
return 1;
}
else
{
oldCursor_ = hCursor;
}
//
// Get cursor info.
//
// logTest("Poller::Cursor","hCursor [%xH] GetCursor [%xH].\n", hCursor, GetCursor());
ICONINFO iconInfo;
if (GetIconInfo(hCursor, &iconInfo) == 0)
{
logTest("Poller::Cursor","GetIconInfo() failed. Error[%d].", (unsigned int)GetLastError());
LocalFree(&iconInfo);
// return -1;
}
BOOL isColorCursor = FALSE;
if (iconInfo.hbmColor != NULL)
{
isColorCursor = TRUE;
}
if (iconInfo.hbmMask == NULL)
{
logTest("Poller::Cursor","Cursor bitmap handle is NULL.\n");
return -1;
}
//
// Check bitmap info for the cursor
//
BITMAP bmMask;
if (!GetObject(iconInfo.hbmMask, sizeof(BITMAP), (LPVOID)&bmMask))
{
logTest("Poller::Cursor","GetObject() for bitmap failed.\n");
DeleteObject(iconInfo.hbmMask);
LocalFree(&bmMask);
return -1;
}
if (bmMask.bmPlanes != 1 || bmMask.bmBitsPixel != 1)
{
logTest("Poller::Cursor","Incorrect data in cursor bitmap.\n");
LocalFree(&bmMask);
DeleteObject(iconInfo.hbmMask);
return -1;
}
// Get monochrome bitmap data for cursor
// NOTE: they say we should use GetDIBits() instead of GetBitmapBits().
mBits = new BYTE[bmMask.bmWidthBytes * bmMask.bmHeight];
if (mBits == NULL)//Data bitmap
{
DeleteObject(iconInfo.hbmMask);
DestroyCursor(cursorInfo.hCursor);
LocalFree(&iconInfo);
LocalFree(&bmMask);
delete[] mBits;
return -1;
}
BOOL success = GetBitmapBits(iconInfo.hbmMask, bmMask.bmWidthBytes * bmMask.bmHeight, mBits);
if (!success)
{
logTest("Poller::Cursor","GetBitmapBits() failed.\n");
delete[] mBits;
return -1;
}
andBits = mBits;
long width = bmMask.bmWidth;
long height = (isColorCursor) ? bmMask.bmHeight : bmMask.bmHeight/2;
//
// The bitmask is formatted so that the upper half is
// the icon AND bitmask and the lower half is the icon XOR bitmask.
//
if (!isColorCursor)
{
xorBits = andBits + bmMask.bmWidthBytes * height;
/* logTest("Poller::Cursor","no color widthB[%ld] width[%ld] height[%ld] totByte[%ld] mbits[%ld].\n",
bmMask.bmWidthBytes,width,height,success,bmMask.bmHeight * bmMask.bmWidthBytes);*/
if (xCursor_ > 0)
{
XFreeCursor(display_, xCursor_);
}
xCursor_ = createCursor(wnd, vis, (unsigned int)iconInfo.xHotspot, (unsigned int)iconInfo.yHotspot,
width, height, (unsigned char *)xorBits, (unsigned char *)andBits);
XDefineCursor(display_, wnd, xCursor_);
}
delete []mBits;
DeleteObject(iconInfo.hbmMask);
LocalFree(&bmMask);
DestroyCursor(cursorInfo.hCursor);
LocalFree(&iconInfo);
return success;
}
unsigned char Poller::specialKeys(unsigned int keysym, unsigned int state, int pressed)
{
return 0;
}
void Poller::ensureServerModifiers(keyTranslation tr, int *lengthArrayINPUT)
{
return;
}
void Poller::restoreServerModifiers(UINT scancode)
{
keyTranslation dummy;
int lengthArrayINPUT = 0;
if (isModifier(scancode) == 1)
{
return;
}
dummy.scancode = 0;
dummy.modifiers = savedServerModifierState_;
ensureServerModifiers(dummy, &lengthArrayINPUT);
if (sendInput(0, 0, &lengthArrayINPUT) == 0)
{
logTest("Poller::restoreServerModifiers", "lengthArrayINPUT [%d]", lengthArrayINPUT);
}
}
int Poller::updateShadowFrameBuffer(void)
{
return 1;
}
void Poller::addToKeymap(char *keyname, unsigned char scancode, unsigned short modifiers, char *mapname)
{
return;
}
FILE *Poller::xkeymapOpen(char *filename)
{
return NULL;
}
int Poller::xkeymapRead(char *mapname)
{
return 1;
}
void Poller::xkeymapInit(char *keyMapName)
{
return;
}
keyTranslation Poller::xkeymapTranslateKey(unsigned int keysym, unsigned int keycode,
unsigned int state)
{
keyTranslation tr = { 0, 0 };
return tr;
}
unsigned char Poller::getKeyState(unsigned int state, unsigned int keysym)
{
return 0;
}
char *Poller::getKsname(unsigned int keysym)
{
char *ksname = NULL;
return ksname;
}
//
// Routine used to fool Winlogon into thinking CtrlAltDel was pressed
//
char Poller::simulateCtrlAltDel(void)
{
HDESK oldDesktop = GetThreadDesktop(GetCurrentThreadId());
//
// Switch into the Winlogon desktop.
//
if (selectDesktopByName("Winlogon") == 0)
{
logTest("SimulateCtrlAltDelThreadFn","Failed to select logon desktop.");
return 0;
}
logTest("SimulateCtrlAltDelThreadFn","Generating ctrl-alt-del.");
//
// Winlogon uses hotkeys to trap Ctrl-Alt-Del.
//
PostMessage(HWND_BROADCAST, WM_HOTKEY, 0, MAKELONG(MOD_ALT | MOD_CONTROL, VK_DELETE));
//
// Switch back to our original desktop.
//
if (oldDesktop != NULL)
{
selectDesktop(oldDesktop);
}
return 1;
}
// Switches the current thread into a different desktop by desktop handle
// This call takes care of all the evil memory management involved
char Poller::selectDesktop(HDESK newDesktop)
{
//
// Only on NT.
//
if (isWinNT())
{
HDESK oldDesktop = GetThreadDesktop(GetCurrentThreadId());
DWORD dummy;
char newName[256];
if (GetUserObjectInformation(newDesktop, UOI_NAME, &newName, 256, &dummy) == 0)
{
logDebug("Poller::selectDesktop","GetUserObjectInformation() failed. Error[%lu].", GetLastError());
return 0;
}
logTest("Poller::selectDesktop","New Desktop to [%s] (%x) from (%x).",
newName, (unsigned int)newDesktop, (unsigned int)oldDesktop);
//
// Switch the desktop.
//
if(SetThreadDesktop(newDesktop) == 0)
{
logDebug("Poller::SelectDesktop","Unable to SetThreadDesktop(), Error=%lu.", GetLastError());
return 0;
}
//
// Switched successfully - destroy the old desktop.
//
if (CloseDesktop(oldDesktop) == 0)
{
logDebug("Poller::selectHdesk","Failed to close old desktop (%x), Error=%lu.",
(unsigned int)oldDesktop, GetLastError());
return 0;
}
}
return 1;
}
//
// Switches the current thread into a different desktop, by name
// Calling with a valid desktop name will place the thread in that desktop.
// Calling with a NULL name will place the thread in the current input desktop.
//
char Poller::selectDesktopByName(char *name)
{
//
// Only on NT.
//
if (isWinNT())
{
HDESK desktop;
if (name != NULL)
{
//
// Attempt to open the named desktop.
//
desktop = OpenDesktop(name, 0, FALSE,
DESKTOP_CREATEMENU | DESKTOP_CREATEWINDOW |
DESKTOP_ENUMERATE | DESKTOP_HOOKCONTROL |
DESKTOP_WRITEOBJECTS | DESKTOP_READOBJECTS |
DESKTOP_SWITCHDESKTOP | GENERIC_WRITE);
}
else
{
//
// Open the input desktop.
//
desktop = OpenInputDesktop(0, FALSE,
DESKTOP_CREATEMENU | DESKTOP_CREATEWINDOW |
DESKTOP_ENUMERATE | DESKTOP_HOOKCONTROL |
DESKTOP_WRITEOBJECTS | DESKTOP_READOBJECTS |
DESKTOP_SWITCHDESKTOP | GENERIC_WRITE);
}
if (desktop == NULL)
{
logDebug("Poller::selectDesktopByName","Unable to open desktop, Error=%lu.", GetLastError());
return 0;
}
//
// Switch to the new desktop
//
if (selectDesktop(desktop) == 0)
{
//
// Failed to enter the new desktop, so free it!
//
logDebug("Poller::selectDesktopByName","Failed to select desktop.");
if (CloseDesktop(desktop) == 0)
{
logDebug("Poller::selectDesktopByName","Failed to close desktop, Error=%lu.", GetLastError());
return 0;
}
}
return 1;
}
return (name == NULL);
}
void Poller::platformOS()
{
OSVERSIONINFO osversioninfo;
osversioninfo.dwOSVersionInfoSize = sizeof(osversioninfo);
//
// Get the current OS version.
//
if (GetVersionEx(&osversioninfo) == 0)
{
platformID_ = 0;
}
platformID_ = osversioninfo.dwPlatformId;
//
// versionMajor = osversioninfo.dwMajorVersion;
// versionMinor = osversioninfo.dwMinorVersion;
//
}
char Poller::checkDesktop()
{
//
// Only on NT.
//
if (isWinNT())
{
//
// Get the input and thread desktops.
//
HDESK desktop = GetThreadDesktop(GetCurrentThreadId());
HDESK inputDesktop = OpenInputDesktop(0, FALSE,
DESKTOP_CREATEMENU | DESKTOP_CREATEWINDOW |
DESKTOP_ENUMERATE | DESKTOP_HOOKCONTROL |
DESKTOP_WRITEOBJECTS | DESKTOP_READOBJECTS |
DESKTOP_SWITCHDESKTOP | GENERIC_WRITE);
if (inputDesktop == NULL)
{
return 0;
}
DWORD dummy;
char desktopName[256];
char inputName[256];
if (GetUserObjectInformation(desktop, UOI_NAME, &desktopName, 256, &dummy) == 0)
{
if (CloseDesktop(inputDesktop) == 0)
{
logDebug("Poller::checkDesktop", "Failed to close desktop, Error[%d].", (unsigned int)GetLastError());
return 0;
}
}
if (GetUserObjectInformation(inputDesktop, UOI_NAME, &inputName, 256, &dummy) == 0)
{
if (CloseDesktop(inputDesktop) == 0)
{
logDebug("Poller::checkDesktop", "Failed to close input desktop, Error[%d].", (unsigned int)GetLastError());
return 0;
}
}
if (strcmp(desktopName, inputName) != 0)
{
//
// Switch to new desktop.
//
selectDesktop(inputDesktop);
}
if (CloseDesktop(desktop) == 0)
{
logDebug("Poller::checkDesktop", "Failed to close input desktop, Error[%d].", (unsigned int)GetLastError());
return 0;
}
if (CloseDesktop(inputDesktop) == 0)
{
logDebug("Poller::checkDesktop", "Failed to close input desktop, Error[%d].", (unsigned int)GetLastError());
return 0;
}
}
return 1;
}
unsigned char Poller::isModifier(UINT scancode)
{
return 0;
}
void Poller::updateModifierState(UINT scancode, unsigned char pressed)
{
return;
}
Cursor Poller::createCursor(Window wnd, Visual *vis,unsigned int x, unsigned int y,
int width, int height, unsigned char *xormask, unsigned char *andmask)
{
Pixmap maskglyph, cursorglyph;
XColor bg, fg;
Cursor xcursor;
unsigned char *cursor;
unsigned char *mask, *pmask, *pcursor, tmp;
int scanline, offset;
scanline = (width + 7) / 8;
offset = scanline * height;
pmask = andmask;
pcursor = xormask;
for (int i = 0; i < offset; i++)
{
//
// The pixel is black if both the bit of andmask and xormask is one.
//
tmp = *pcursor & *pmask;
*pcursor ^= tmp;
*pmask ^= tmp;
*pmask = ~(*pmask);
pmask++;
pcursor++;
}
cursor = new unsigned char[offset];
memcpy(cursor, xormask, offset);
mask = new unsigned char[offset];
memcpy(mask, andmask, offset);
fg.red = fg.blue = fg.green = 0xffff;
bg.red = bg.blue = bg.green = 0x0000;
fg.flags = bg.flags = DoRed | DoBlue | DoGreen;
cursorglyph = createGlyph(wnd, vis, width, height, cursor);
maskglyph = createGlyph(wnd, vis, width, height, mask);
xcursor = XCreatePixmapCursor(display_, cursorglyph, maskglyph, &fg, &bg, x, y);
XFreePixmap(display_, maskglyph);
XFreePixmap(display_, cursorglyph);
delete[]mask;
delete[]cursor;
return xcursor;
}
Pixmap Poller::createGlyph(Window wnd, Visual *visual, int width, int height, unsigned char *data)
{
XImage *image;
Pixmap bitmap;
int scanline;
GC glyphGC;
scanline = (width + 7) / 8;
bitmap = XCreatePixmap(display_, wnd, width, height, 1);
glyphGC = XCreateGC(display_, bitmap, 0, NULL);
image = XCreateImage(display_, visual, 1, ZPixmap, 0, (char *)data, width, height, 8, scanline);
image->byte_order = 1; // MSBFirst -- LSBFirst = 0
image->bitmap_bit_order = 1;
XInitImage(image);
/* logTest("Poller::createGlyph","XPutImage on pixmap %d,%d,%d,%d.\n",
0, 0, width, height);*/
XPutImage(display_, bitmap, glyphGC, image, 0, 0, 0, 0, width, height);
XFree(image);
return bitmap;
}
#endif /* defined(__CYGWIN32__) || defined(WIN32) */
/**************************************************************************/
/* */
/* Copyright (c) 2001, 2007 NoMachine, http://www.nomachine.com/. */
/* */
/* NXCOMPSHAD, NX protocol compression and NX extensions to this software */
/* are copyright of NoMachine. Redistribution and use of the present */
/* software is allowed according to terms specified in the file LICENSE */
/* which comes in the source distribution. */
/* */
/* Check http://www.nomachine.com/licensing.html for applicability. */
/* */
/* NX and NoMachine are trademarks of Medialogic S.p.A. */
/* */
/* All rights reserved. */
/* */
/**************************************************************************/
#ifdef __CYGWIN32__
#ifndef Win32Poller_H
#define Win32Poller_H
//#include <X11/X.h>
#include <Windows.h>
#include <wingdi.h>
#include <winable.h>
#include <winuser.h>
#define CAPTUREBLT 0x40000000
#define KEYEVENTF_SCANCODE 0x00000008
#define MAPVK_VSC_TO_VK_EX 3
//
// The CAPTUREBLT is a raster operation used
// in bit blit transfer.
//
// Using this operation includes any windows
// that are layered on top of your window in
// the resulting image. By default, the image
// only contains your window.
//
#include "Core.h"
typedef struct _keyTranslation
{
unsigned char scancode;
unsigned short modifiers;
}keyTranslation;
class Poller : public CorePoller
{
public:
Display *display_;
keyTranslation *keymap_;
unsigned char keymapLoaded_;
int minKeycode_;
Poller(Input *, Display *display, int = 16);
~Poller();
int init();
int updateCursor(Window, Visual*);
private:
int Poller::updateShadowFrameBuffer(void);
void handleKeyboardEvent(Display *, XEvent *);
void addToKeymap(char *keyname, unsigned char scancode, unsigned short modifiers, char *mapname);
int xkeymapRead(char *mapname);
FILE *xkeymapOpen(char *filename);
void xkeymapInit(char *keyMapName);
keyTranslation xkeymapTranslateKey(unsigned int keysym, unsigned int keycode, unsigned int state);
unsigned char getKeyState(unsigned int state, unsigned int keysym);
char *getKsname(unsigned int keysym);
unsigned char specialKeys(unsigned int keysym, unsigned int state, int pressed);
unsigned char toggleSwitch(unsigned char ToggleStateClient, unsigned char ToggleStateServer, UINT scancode,
int *lengthArrayINPUT);
void updateModifierState(UINT, unsigned char);
unsigned char toggleServerState(UINT scancode);
unsigned char keyState(UINT scancode, UINT mapType);
unsigned char keyStateAsync(UINT scancode);
void handleMouseEvent(Display *, XEvent *);
Cursor createCursor(Window wnd, Visual *vis, unsigned int x, unsigned int y, int width,
int height, unsigned char *xormask, unsigned char *andmask);
Pixmap createGlyph(Window wnd, Visual *visual, int width, int height, unsigned char *data);
char isWinNT();
char selectDesktop(HDESK new_desktop);
char selectDesktopByName(char *name);
void platformOS();
char simulateCtrlAltDel(void);
DWORD platformID_;
INPUT *pKey_, *pMouse_;
char *keymapName_;
char *path_;
unsigned char toggleButtonState_;
unsigned short serverModifierState_;
unsigned short savedServerModifierState_;
void ensureServerModifiers(keyTranslation tr, int *lenghtArrayINPUT);
void restoreServerModifiers(UINT scancode);
unsigned char isModifier(UINT scancode);
char sendInput(unsigned char scancode, unsigned char pressed, int *lengthArrayINPUT);
char *getRect(XRectangle);
char checkDesktop();
char *DIBBuffer_;
HCURSOR oldCursor_;
VOID *pDIBbits_;
HDC screenDC_;
HDC memoryDC_;
BITMAPINFO bmi_;
HBITMAP screenBmp_;
Cursor xCursor_;
};
#undef TEST
inline unsigned char Poller::toggleSwitch(unsigned char ToggleStateClient, unsigned char ToggleStateServer,
UINT scancode, int *lengthArrayINPUT)
{
return 1;
}
inline unsigned char Poller::toggleServerState(UINT scancode)
{
return (GetKeyState(MapVirtualKeyEx(scancode, 3, GetKeyboardLayout((DWORD)NULL))) & 0x1);
}
inline unsigned char Poller::keyStateAsync(UINT vKeycode)
{
return GetAsyncKeyState(vKeycode);
}
inline unsigned char Poller::keyState(UINT code, UINT mapType)
{
if (mapType == 0)
{
//
// Virtual Keycode
//
return ((GetKeyState(code) & 0x80) == 0x80);
}
else
{
//
// scancode
//
return ((GetKeyState(MapVirtualKeyEx(code, 3, GetKeyboardLayout((DWORD)NULL))) & 0x80) == 0x80);
}
}
inline char Poller::isWinNT()
{
return (platformID_ == VER_PLATFORM_WIN32_NT);
}
inline char Poller::sendInput(unsigned char scancode, unsigned char pressed, int *lengthArrayINPUT)
{
DWORD keyEvent = 0;
DWORD extended = 0;
if (scancode > 0)
{
if (pressed == 0)
{
keyEvent = KEYEVENTF_KEYUP;
}
if (scancode & 0x80)
{
scancode &= ~0x80;
extended = KEYEVENTF_EXTENDEDKEY;
}
pKey_[*lengthArrayINPUT].ki.wScan = (WORD) scancode;
pKey_[*lengthArrayINPUT].ki.dwFlags = (DWORD) (keyEvent | KEYEVENTF_SCANCODE | extended);
(*lengthArrayINPUT)++;
}
if (*lengthArrayINPUT > 0) {
// FIXME: Remove me.
logTest("Poller::sendInput", "length Input [%d] event: %s", *lengthArrayINPUT,
pressed == 1 ? "KeyPress": "KeyRelease");
if (SendInput(*lengthArrayINPUT, pKey_, sizeof(INPUT)) == 0)
{
logTest("Poller::sendInput", "Failed SendInput, event: %s", pressed == 1 ? "KeyPress": "KeyRelease");
*lengthArrayINPUT = 0;
return 0;
}
*lengthArrayINPUT = 0;
}
return 1;
}
#endif /* Win32Poller_H */
#endif /* __CYGWIN32__ */
/**************************************************************************/
/* */
/* Copyright (c) 2001, 2007 NoMachine, http://www.nomachine.com/. */
/* */
/* NXCOMPSHAD, NX protocol compression and NX extensions to this software */
/* are copyright of NoMachine. Redistribution and use of the present */
/* software is allowed according to terms specified in the file LICENSE */
/* which comes in the source distribution. */
/* */
/* Check http://www.nomachine.com/licensing.html for applicability. */
/* */
/* NX and NoMachine are trademarks of Medialogic S.p.A. */
/* */
/* All rights reserved. */
/* */
/**************************************************************************/
#if !defined(__CYGWIN32__) && !defined(WIN32)
#define PANIC
#define WARNING
#undef TEST
#undef DEBUG
#include <X11/Xlibint.h>
#include <X11/extensions/XTest.h>
#include <X11/keysym.h>
#include <string.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#include "Poller.h"
#include "Logger.h"
#include "Shadow.h"
#define ROUNDUP(nbytes, pad) ((((nbytes) + ((pad)-1)) / (pad)) * ((pad)>>3))
#define TRANSLATE_KEYCODES
Poller::Poller(Input *input, Display *display, int depth) : CorePoller(input, display)
{
logTrace("Poller::Poller");
display_ = NULL;
shadowDisplayName_ = input -> getShadowDisplayName();
tmpBuffer_ = NULL;
xtestExtension_ = -1;
shmExtension_ = -1;
randrExtension_ = -1;
damageExtension_ = -1;
shadowDisplayUid_ = -1;
image_ = NULL;
shminfo_ = NULL;
}
Poller::~Poller()
{
logTrace("Poller::~Poller");
if (shmExtension_ == 1)
{
XShmDetach(display_, shminfo_);
XDestroyImage(image_);
shmdt(shminfo_ -> shmaddr);
shmctl(shminfo_ -> shmid, IPC_RMID, 0);
}
if (shminfo_ != NULL)
{
delete shminfo_;
shminfo_ = NULL;
}
if (display_ != NULL)
{
XCloseDisplay(display_);
}
if (tmpBuffer_ != NULL && shmExtension_ == 1 && damageExtension_ == 1)
{
XFree(tmpBuffer_);
tmpBuffer_ = NULL;
}
}
int Poller::init()
{
logTrace("Poller::init");
if (display_ == NULL)
{
display_ = XOpenDisplay(shadowDisplayName_);
setShadowDisplay(display_);
}
logTest("Poller::init:" ,"Shadow display [%p] name [%s].", (Display *) display_, shadowDisplayName_);
if (display_ == NULL)
{
logTest("Poller::init", "Failed to connect to display [%s].", shadowDisplayName_ ? shadowDisplayName_ : "");
return -1;
}
setRootSize();
logTest("Poller::init", "Screen geometry is [%d, %d] depth is [%d] bpl [%d] bpp [%d].",
width_, height_, depth_, bpl_, bpp_);
xtestInit();
shmInit();
randrInit();
damageInit();
return CorePoller::init();
}
int Poller::updateShadowFrameBuffer(void)
{
if (shmExtension_ == 1)
{
if (XShmGetImage(display_, DefaultRootWindow(display_), image_, 0, 0, AllPlanes) == 0)
{
logDebug("Poller::updateShadowFrameBuffer", "XShmGetImage failed!");
return -1;
}
return 1;
}
else
{
return 0;
}
}
char *Poller::getRect(XRectangle r)
{
logTrace("Poller::getRect");
logDebug("Poller::getRect", "Going to retrive rectangle [%d, %d, %d, %d].",
r.x, r.y, r.width, r.height);
if (shmExtension_ == 1)
{
if (damageExtension_ == 1)
{
image_ -> width = r.width;
image_ -> height = r.height;
image_ -> bytes_per_line = ROUNDUP((image_ -> bits_per_pixel * image_ -> width), image_ -> bitmap_pad);
if (XShmGetImage(display_, DefaultRootWindow(display_), image_, r.x, r.y, AllPlanes) == 0)
{
logDebug("Poller::getRect", "XShmGetImage failed!");
return NULL;
}
tmpBuffer_ = image_ -> data;
}
else
{
image_ -> width = r.width;
image_ -> height = r.height;
image_ -> bytes_per_line = ROUNDUP((image_ -> bits_per_pixel * image_ -> width), image_ -> bitmap_pad);
if (XShmGetImage(display_, DefaultRootWindow(display_), image_, r.x, r.y, AllPlanes) == 0)
{
logDebug("Poller::getRect", "XShmGetImage failed!");
}
tmpBuffer_ = image_ -> data;
}
}
else
{
if (tmpBuffer_)
{
XFree(tmpBuffer_);
tmpBuffer_ = NULL;
}
image_ = XGetImage(display_, DefaultRootWindow(display_), r.x, r.y, r.width, r.height, AllPlanes, ZPixmap);
if (image_ == NULL)
{
logError("Poller::getRect", ESET(ENOMSG));
return NULL;
}
tmpBuffer_ = image_ -> data;
if (image_ -> obdata)
{
XFree(image_ -> obdata);
}
XFree(image_);
}
return tmpBuffer_;
}
void Poller::shmInit(void)
{
int major, minor;
int pixmaps;
logTest("Poller::shmInit", "Added shmExtension_ [%d].", shmExtension_);
if (shmExtension_ >= 0)
{
logDebug("Poller::shmInit", "Called with shared memory already initialized.");
return;
}
if (shmExtension_ < 0 && NXShadowOptions.optionShmExtension == 0)
{
shmExtension_ = 0;
logUser("Poller::shmInit: Disabling use of MIT-SHM extension.\n");
return;
}
if (XShmQueryVersion(display_, &major, &minor, &pixmaps) == 0)
{
logDebug("Poller::shmInit", "MIT_SHM: Shared memory extension not available.");
shmExtension_ = 0;
}
else
{
logDebug("Poller::shmInit", "MIT_SHM: Shared memory extension available.");
if (shminfo_ != NULL)
{
destroyShmImage();
}
shminfo_ = (XShmSegmentInfo* ) new XShmSegmentInfo;
if (shminfo_ == NULL)
{
logError("Poller::shmInit", ESET(ENOMEM));
shmExtension_ = 0;
return;
}
image_ = (XImage *)XShmCreateImage(display_, display_ -> screens[0].root_visual, depth_, ZPixmap,
NULL, shminfo_, width_, height_);
if (image_ == NULL)
{
logError("Poller::shmInit", ESET(ENOMSG));
shmExtension_ = 0;
return;
}
shadowDisplayUid_ = NXShadowOptions.optionShadowDisplayUid;
logDebug("Poller::shmInit", "Master X server uid [%d].", NXShadowOptions.optionShadowDisplayUid);
shminfo_ -> shmid = shmget(IPC_PRIVATE, image_ -> bytes_per_line * image_ -> height, IPC_CREAT | 0666);
if (shminfo_ -> shmid < 0)
{
logDebug("Poller::shmInit", "kernel id error.");
shmExtension_ = 0;
return;
}
logDebug("Poller::shmInit", "Created shm segment with shmid [%d].", shminfo_ -> shmid);
shminfo_ -> shmaddr = (char *)shmat(shminfo_ -> shmid, 0, 0);
if (shminfo_ -> shmaddr < 0)
{
logWarning("Poller::shmInit", "Couldn't attach to shm segment.");
}
logDebug("Poller::shmInit", "shminfo_ -> shmaddr [%p].", shminfo_ -> shmaddr);
image_ -> data = shminfo_ -> shmaddr;
shminfo_ -> readOnly = 0;
if (XShmAttach(display_, shminfo_) == 0)
{
logDebug("Poller::shmInit", "XShmAttach failed.");
shmExtension_ = 0;
return;
}
//
// Mark the shm segment to be destroyed after
// the last process detach. Let the X server
// complete the X_ShmAttach request, before.
//
XSync(display_, 0);
struct shmid_ds ds;
shmctl(shminfo_ -> shmid, IPC_STAT, &ds);
if (shadowDisplayUid_ != -1)
{
ds.shm_perm.uid = (ushort) shadowDisplayUid_;
}
else
{
logWarning("Poller::shmInit", "Couldn't set uid for shm segment.");
}
ds.shm_perm.mode = 0600;
shmctl(shminfo_ -> shmid, IPC_SET, &ds);
shmctl(shminfo_ -> shmid, IPC_STAT, &ds);
shmctl(shminfo_ -> shmid, IPC_RMID, 0);
logDebug("Poller::shmInit", "Number of attaches to shm segment [%d] are [%d].\n",
shminfo_ -> shmid, (int) ds.shm_nattch);
if (ds.shm_nattch > 2)
{
logWarning("Poller::shmInit", "More than two attaches to the shm segment.");
destroyShmImage();
shmExtension_ = 0;
return;
}
shmExtension_ = 1;
}
}
void Poller::handleKeyboardEvent(Display *display, XEvent *event)
{
if (xtestExtension_ == 0 || display_ == 0)
{
return;
}
logTest("Poller::handleKeyboardEvent", "Handling event at [%p]", event);
//
// Use keysyms to translate keycodes across different
// keyboard models. Unuseful when both keyboards have
// same keycodes (e.g. both are pc keyboards).
//
#ifdef TRANSLATE_KEYCODES
KeyCode keycode = XKeysymToKeycode(display_, XK_A);
if (XKeysymToKeycode(event -> xkey.display, XK_A) != keycode)
{
KeySym keysym = XKeycodeToKeysym(event -> xkey.display, event -> xkey.keycode, 0);
if (keysym == XK_Mode_switch || keysym == XK_ISO_Level3_Shift)
{
logUser("Poller::handleKeyboardEvent: keysym [%x].\n", (unsigned int)keysym);
if (XKeycodeToKeysym(display_, 113, 0) == XK_ISO_Level3_Shift ||
(XKeycodeToKeysym(display_, 124, 0) == XK_ISO_Level3_Shift))
{
event -> xkey.keycode = 113;
}
else
{
event -> xkey.keycode = XKeysymToKeycode(display_, XK_Mode_switch);
}
logUser("Poller::handleKeyboardEvent: keycode translated to [%x].\n", (unsigned int)event -> xkey.keycode);
}
else
{
event -> xkey.keycode = XKeysymToKeycode(display_, keysym);
}
}
#endif // TRANSLATE_KEYCODES
if (event -> type == KeyPress)
{
XTestFakeKeyEvent(display_, event -> xkey.keycode, 1, 0);
}
else if (event -> type == KeyRelease)
{
XTestFakeKeyEvent(display_, event -> xkey.keycode, 0, 0);
}
}
void Poller::handleMouseEvent(Display *display, XEvent *event)
{
if (xtestExtension_ == 0 || display_ == 0)
{
return;
}
if (event -> type == MotionNotify)
{
XTestFakeMotionEvent(display_, 0, event -> xmotion.x, event -> xmotion.y, 0);
}
else if (event -> type == ButtonPress)
{
XTestFakeButtonEvent(display_, event -> xbutton.button, True, 0);
}
else if (event -> type == ButtonRelease)
{
XTestFakeButtonEvent(display_, event -> xbutton.button, False, 0);
}
XFlush(display_);
}
void Poller::setRootSize(void)
{
width_ = WidthOfScreen(DefaultScreenOfDisplay(display_));
height_ = HeightOfScreen(DefaultScreenOfDisplay(display_));
depth_ = DefaultDepth(display_, DefaultScreen(display_));
if (depth_ == 8) bpp_ = 1;
else if (depth_ == 16) bpp_ = 2;
else bpp_ = 4;
bpl_ = width_ * bpp_;
}
void Poller::destroyShmImage(void)
{
XShmDetach(display_, shminfo_);
XDestroyImage(image_);
image_ = NULL;
shmdt(shminfo_ -> shmaddr);
shmctl(shminfo_ -> shmid, IPC_RMID, 0);
delete shminfo_;
shminfo_ = NULL;
}
void Poller::xtestInit(void)
{
int eventBase;
int errorBase;
int versionMajor;
int versionMinor;
int result;
xtestExtension_ = 0;
result = XTestQueryExtension(display_, &eventBase, &errorBase, &versionMajor, &versionMinor);
if (result == 0)
{
xtestExtension_ = 0;
logWarning("Poller::xtestInit", "Failed while querying for XTEST extension.");
}
else
{
logDebug("Poller::xtestInit", "XTEST version %d.%d.", versionMajor, versionMinor);
xtestExtension_ = 1;
}
//
// Make this client impervious to grabs.
//
if (xtestExtension_ == 1)
{
XTestGrabControl(display_, 1);
}
}
void Poller::randrInit(void)
{
int randrEventBase;
int randrErrorBase;
randrExtension_ = 0;
XRRSelectInput(display_, DefaultRootWindow(display_), RRScreenChangeNotifyMask);
if (XRRQueryExtension(display_, &randrEventBase, &randrErrorBase) == 0)
{
#ifdef PANIC
fprintf(stderr, "nxagentShadowInit: Randr extension not supported on this display.\n");
#endif
}
randrEventBase_ = randrEventBase;
randrExtension_ = 1;
return;
}
void Poller::damageInit(void)
{
int damageMajorVersion = 0;
int damageMinorVersion = 0;
int damageEventBase = 0;
int damageErrorBase = 0;
if (damageExtension_ >= 0)
{
logDebug("Poller::damageInit", "Called with damage already initialized.");
}
if (damageExtension_ == 0)
{
logDebug("Poller::damageInit", "Damage disabled. Skip initialization.");
return;
}
if (damageExtension_ < 0 && NXShadowOptions.optionDamageExtension == 0)
{
damageExtension_ = 0;
logUser("Poller::damageInit: Disabling use of DAMAGE extension.\n");
return;
}
damageExtension_ = 0;
mirrorChanges_ = 0;
if (XDamageQueryExtension(display_, &damageEventBase, &damageErrorBase) == 0)
{
logUser("Poller::damageInit: DAMAGE not supported.\n");
return;
}
#ifdef DEBUG
else
{
fprintf(stderr, "Poller::damageInit: DAMAGE supported. "
"Event base [%d] error base [%d].\n", damageEventBase, damageErrorBase);
}
#endif
damageEventBase_ = damageEventBase;
if (XDamageQueryVersion(display_, &damageMajorVersion, &damageMinorVersion) == 0)
{
logWarning("Poller::damageInit", "Error on querying DAMAGE version.\n");
damageExtension_ = 0;
return;
}
#ifdef DEBUG
else
{
fprintf(stderr, "Poller::damageInit: DAMAGE version %d.%d.\n",
damageMajorVersion, damageMinorVersion);
}
#endif
damage_ = XDamageCreate(display_, DefaultRootWindow(display_), XDamageReportRawRectangles);
damageExtension_= 1;
mirror_ = 1;
return;
}
void Poller::getEvents(void)
{
XEvent X;
if (damageExtension_ == 1)
{
XDamageSubtract(display_, damage_, None, None);
}
XSync(display_, 0);
while (XCheckIfEvent(display_, &X, anyEventPredicate, NULL) == 1)
{
if (randrExtension_ == 1 && (X.type == randrEventBase_ + RRScreenChangeNotify || X.type == ConfigureNotify))
{
XRRUpdateConfiguration (&X);
handleRRScreenChangeNotify(&X);
continue;
}
if (damageExtension_ == 1 && X.type == damageEventBase_ + XDamageNotify)
{
handleDamageNotify(&X);
}
}
if (damageExtension_ == 1)
{
updateDamagedAreas();
}
XFlush(display_);
}
void Poller::handleRRScreenChangeNotify(XEvent *X)
{
return;
}
void Poller::handleDamageNotify(XEvent *X)
{
XDamageNotifyEvent *e = (XDamageNotifyEvent *) X;
//
// e->drawable is the window ID of the damaged window
// e->geometry is the geometry of the damaged window
// e->area is the bounding rect for the damaged area
// e->damage is the damage handle returned by XDamageCreate()
//
#ifdef DEBUG
fprintf(stderr, "handleDamageNotify: drawable [%d] damage [%d] geometry [%d][%d][%d][%d] area [%d][%d][%d][%d].\n",
(int) e -> drawable, (int) e -> damage, e -> geometry.x, e -> geometry.y,
e -> geometry.width, e -> geometry.height, e -> area.x, e -> area.y,
e -> area.width, e -> area.height);
#endif
XRectangle rectangle = {e -> area.x, e -> area.y, e -> area.width, e -> area.height};
XUnionRectWithRegion(&rectangle, lastUpdatedRegion_, lastUpdatedRegion_);
mirrorChanges_ = 1;
return;
}
void Poller::updateDamagedAreas(void)
{
if (shmExtension_ == 1)
{
BOX *boxPtr;
XRectangle rectangle;
int i;
int y;
for (i = 0; i < lastUpdatedRegion_ -> numRects; i++)
{
boxPtr = lastUpdatedRegion_ -> rects + i;
image_ -> width = boxPtr -> x2 - boxPtr -> x1;
image_ -> height = boxPtr -> y2 - boxPtr -> y1;
image_ -> bytes_per_line = ROUNDUP((image_ -> bits_per_pixel * image_ -> width), image_ -> bitmap_pad);
if (XShmGetImage(display_, DefaultRootWindow(display_), image_, boxPtr -> x1, boxPtr -> y1, AllPlanes) == 0)
{
logDebug("Poller::getRect", "XShmGetImage failed!");
return;
}
rectangle.height = 1;
rectangle.width = image_ -> width;
rectangle.x = boxPtr -> x1;
rectangle.y = boxPtr -> y1;
for (y = 0; y < image_ -> height; y++)
{
update(image_ -> data + y * image_ -> bytes_per_line, rectangle);
rectangle.y++;
}
}
}
return;
}
int anyEventPredicate(Display *display, XEvent *event, XPointer parameter)
{
return 1;
}
#endif /* !defined(__CYGWIN32__) && !defined(WIN32) */
/**************************************************************************/
/* */
/* Copyright (c) 2001, 2007 NoMachine, http://www.nomachine.com/. */
/* */
/* NXCOMPSHAD, NX protocol compression and NX extensions to this software */
/* are copyright of NoMachine. Redistribution and use of the present */
/* software is allowed according to terms specified in the file LICENSE */
/* which comes in the source distribution. */
/* */
/* Check http://www.nomachine.com/licensing.html for applicability. */
/* */
/* NX and NoMachine are trademarks of Medialogic S.p.A. */
/* */
/* All rights reserved. */
/* */
/**************************************************************************/
#ifndef X11Poller_H
#define X11Poller_H
#include <X11/Xlib.h>
#include <X11/extensions/XShm.h>
#include <X11/extensions/Xdamage.h>
#include <X11/extensions/Xrandr.h>
#include "Core.h"
class Poller : public CorePoller
{
public:
Poller(Input *, Display *display, int = 0);
~Poller();
int init();
void setRootSize();
void destroyShmImage();
void getEvents(void);
private:
Display *display_;
char *shadowDisplayName_;
int shadowDisplayUid_;
char *tmpBuffer_;
char xtestExtension_;
char shmExtension_;
char randrExtension_;
int randrEventBase_;
char damageExtension_;
int damageEventBase_;
Damage damage_;
Region repair_;
char damageChanges_;
XShmSegmentInfo *shminfo_;
XImage *image_;
int updateShadowFrameBuffer(void);
char *getRect(XRectangle);
void handleKeyboardEvent(Display *display, XEvent *);
void handleMouseEvent(Display *, XEvent *);
void xtestInit(void);
void shmInit(void);
void randrInit(void);
void damageInit(void);
void handleRRScreenChangeNotify(XEvent *);
void handleDamageNotify(XEvent *);
void updateDamagedAreas(void);
};
int anyEventPredicate(Display *display, XEvent *event, XPointer parameter);
#endif /* X11Poller_H */
This source diff could not be displayed because it is too large. You can view the blob instead.
dnl Process this file with autoconf to produce a configure script.
dnl Prolog
AC_INIT(Shadow.h)
AC_PREREQ(2.13)
dnl Reset default compilation flags.
CXXFLAGS="-O3"
CPPFLAGS="-O3"
dnl Reset default linking directives.
LIBSTATIC=""
LIBSHARED=""
dnl Prefer headers and libraries from nx-X11, if present.
if test -d "../nx-X11/exports/include" ; then
CXXFLAGS="$CXXFLAGS -I../nx-X11/exports/include"
LIBS="$LIBS -L../nx-X11/exports/lib"
fi
dnl Check whether --with-ipaq was given.
if test "${with_ipaq}" = yes; then
echo -e "enabling IPAQ configuration"
CXX="arm-linux-c++"
CC="arm-linux-gcc"
unset ac_cv_prog_armcxx
unset ac_cv_prog_armcc
unset ac_cv_prog_CXXCPP
AC_CHECK_PROG([armcxx],["$CXX"],[yes],[no],[$PATH])
AC_CHECK_PROG([armcc],["$CC"],[yes],[no],[$PATH])
if test $armcxx = "yes" && test $armcc = "yes" ; then
ac_cv_prog_CXX="$CXX"
ac_cv_prog_CC="$CC"
else
AC_MSG_ERROR(installation or configuration problem: I cannot find compiler for arm-linux)
fi
else
unset ac_cv_prog_CXX
unset ac_cv_prog_CC
unset ac_cv_prog_CXXCPP
fi
dnl Check for programs.
AC_PROG_CXX
AC_PROG_CC
AC_LANG_CPLUSPLUS
dnl Check whether option -Wno-deprecated
dnl is needed by GCC compiler.
AC_MSG_CHECKING([whether compiler needs -Wno-deprecated])
gcc_version=`${CC} --version | grep 'gcc (GCC) [[3-4]].' | head -n 1`
case "${gcc_version}" in
gcc*)
AC_MSG_RESULT([yes])
CXXFLAGS="$CXXFLAGS -Wno-deprecated"
CPPFLAGS="$CPPFLAGS -Wno-deprecated"
;;
*)
AC_MSG_RESULT([no])
;;
esac
AC_MSG_CHECKING([whether compiler accepts -Wmissing-declarations and -Wnested-externs])
gcc_version=`${CC} --version | grep 'gcc (GCC) [[3-4]].' | head -n 1`
case "${gcc_version}" in
gcc*)
AC_MSG_RESULT([no])
;;
*)
AC_MSG_RESULT([yes])
CXXFLAGS="$CXXFLAGS -Wmissing-declarations -Wnested-externs"
CPPFLAGS="$CPPFLAGS -Wmissing-declarations -Wnested-externs"
;;
esac
dnl Check for BSD compatible install.
AC_PROG_INSTALL
dnl Check for extra header files.
AC_PATH_XTRA
dnl Custom addition.
ac_help="$ac_help
--with-symbols give -g flag to compiler to produce debug symbols
--with-info define INFO at compile time to get basic log output
--with-valgrind clean up allocated buffers to avoid valgrind warnings
--with-version use this version for produced libraries
--with-static-jpeg enable static linking of JPEG library
--with-static-z enable static linking of Z library"
dnl Check to see if we're running under Cygwin32.
AC_DEFUN(nxconf_CYGWIN32,
[AC_CACHE_CHECK(for Cygwin32 environment, nxconf_cv_cygwin32,
[AC_TRY_COMPILE(,[return __CYGWIN32__;],
nxconf_cv_cygwin32=yes, nxconf_cv_cygwin32=no)
rm -f conftest*])
CYGWIN32=
test "$nxconf_cv_cygwin32" = yes && CYGWIN32=yes])
nxconf_CYGWIN32
dnl Cygwin32 requires the stdc++ library explicitly linked.
if test "$CYGWIN32" = yes; then
LIBS="$LIBS -lstdc++ -lcygipc -lgdi32"
fi
dnl Check for Darwin environment.
AC_DEFUN(nxconf_DARWIN,
[AC_CACHE_CHECK(for Darwin environment, nxconf_cv_darwin,
[AC_TRY_COMPILE(,[return __APPLE__;],
nxconf_cv_darwin=yes, nxconf_cv_darwin=no)
rm -f conftest*])
DARWIN=
test "$nxconf_cv_darwin" = yes && DARWIN=yes])
nxconf_DARWIN
dnl Check to see if we're running under Solaris.
AC_DEFUN(nxconf_SUN,
[AC_CACHE_CHECK(for Solaris environment, nxconf_cv_sun,
[AC_TRY_COMPILE(,[return __sun;],
nxconf_cv_sun=yes, nxconf_cv_sun=no)
rm -f conftest*])
SUN=
test "$nxconf_cv_sun" = yes && SUN=yes])
nxconf_SUN
dnl Check to see if we're running under FreeBSD.
AC_DEFUN(nxconf_FreeBSD,
[AC_CACHE_CHECK(for FreeBSD environment, nxconf_cv_freebsd,
[AC_TRY_COMPILE(,[return __FreeBSD__;],
nxconf_cv_freebsd=yes, nxconf_cv_freebsd=no)
rm -f conftest*])
FreeBSD=
test "$nxconf_cv_freebsd" = yes && FreeBSD=yes])
nxconf_FreeBSD
dnl Build PIC libraries.
if test "$CYGWIN32" != yes -a "$DARWIN" != yes; then
CXXFLAGS="$CXXFLAGS -fPIC"
CFLAGS="$CFLAGS -fPIC"
fi
dnl Solaris requires the socket and gcc_s libs explicitly linked.
dnl Note also that headers from default /usr/openwin/include/X11
dnl cause a warning due to pragma in Xmd.h.
if test "$SUN" = yes; then
LIBS="$LIBS -L/usr/sfw/lib -lsocket "
CXXFLAGS="$CXXFLAGS -I/usr/sfw/include"
fi
dnl On FreeBSD search libraries and includes under /usr/local.
if test "$FreeBSD" = yes; then
LIBS="$LIBS -L/usr/local/lib"
CXXFLAGS="$CXXFLAGS -I/usr/local/include"
fi
dnl Under Darwin we don't have support for -soname option and
dnl we need the -bundle flag. Under Solaris, instead, we need
dnl the options -G -h.
if test "$DARWIN" = yes; then
LDFLAGS="$LDFLAGS -bundle"
elif test "$SUN" = yes; then
LDFLAGS="$LDFLAGS -G -h \$(LIBLOAD)"
else
LDFLAGS="$LDFLAGS -Wl,-soname,\$(LIBLOAD)"
fi
dnl Check to see if in_addr_t is defined.
dnl Could use a specific configure test.
AC_DEFUN(nxconf_INADDRT,
[AC_CACHE_CHECK(for in_addr_t, nxconf_cv_inaddrt,
[AC_TRY_COMPILE([#include <netinet/in.h>],[in_addr_t t; t = 1; return t;],
nxconf_cv_inaddrt=yes, nxconf_cv_inaddrt=no)
rm -f conftest*])
INADDRT=
test "$nxconf_cv_inaddrt" = yes && INADDRT=yes])
nxconf_INADDRT
dnl If in_addr_t is not defined use unsigned int.
if test "$INADDRT" != yes ; then
echo -e "using unsigned int for type in_addr_t"
CXXFLAGS="$CXXFLAGS -DIN_ADDR_T=unsigned"
CPPFLAGS="$CPPFLAGS -DIN_ADDR_T=unsigned"
else
CXXFLAGS="$CXXFLAGS -DIN_ADDR_T=in_addr_t"
CPPFLAGS="$CPPFLAGS -DIN_ADDR_T=in_addr_t"
fi
dnl Check whether --with-version was given.
AC_SUBST(LIBVERSION)
AC_SUBST(VERSION)
if test "${with_version}" = yes; then
VERSION=${ac_option}
else
VERSION=`cat VERSION`
fi
echo -e "compiling version ${VERSION}"
LIBVERSION=`echo ${VERSION} | cut -d '.' -f 1`
CXXFLAGS="$CXXFLAGS -DVERSION=\\\"${VERSION}\\\""
CPPFLAGS="$CPPFLAGS -DVERSION=\\\"${VERSION}\\\""
dnl Finally compose the LIB variable.
if test "$DARWIN" = yes ; then
LIBS="$LIBS $LIBSTATIC $LIBSHARED"
elif test "$SUN" = yes ; then
LIBS="$LIBS $LIBSTATIC $LIBSHARED"
else
LIBS="$LIBS $LIBSTATIC -shared $LIBSHARED"
fi
dnl Check whether --with-symbols or --without-symbols was
dnl given and set the required optimization level.
if test "${with_symbols}" = yes; then
echo -e "enabling production of debug symbols"
CXXFLAGS="-g $CXXFLAGS"
CPPFLAGS="-g $CPPFLAGS"
else
echo -e "disabling production of debug symbols"
fi
dnl Check whether --with-info or --without-info was given.
if test "${with_info}" = yes; then
echo -e "enabling info output in the log file"
CXXFLAGS="$CXXFLAGS -DINFO"
CPPFLAGS="$CPPFLAGS -DINFO"
else
echo -e "disabling info output in the log file"
fi
dnl Check whether --with-valgrind or --without-valgrind was given.
if test "${with_valgrind}" = yes; then
echo -e "enabling valgrind memory checker workarounds"
CXXFLAGS="$CXXFLAGS -DVALGRIND"
CPPFLAGS="$CPPFLAGS -DVALGRIND"
else
echo -e "disabling valgrind memory checker workarounds"
fi
dnl Find makedepend somewhere.
AC_SUBST(MAKEDEPEND)
if test -x "../nx-X11/config/makedepend/makedepend" ; then
MAKEDEPEND=../nx-X11/config/makedepend/makedepend
else
if test -x "/usr/X11R6/bin/makedepend" ; then
MAKEDEPEND=/usr/X11R6/bin/makedepend
else
if test -x "/usr/openwin/bin/makedepend" ; then
MAKEDEPEND=/usr/openwin/bin/makedepend
else
MAKEDEPEND=makedepend
fi
fi
fi
AC_OUTPUT(Makefile)
#! /bin/sh
#
# install - install a program, script, or datafile
# This comes from X11R5.
#
# Calling this script install-sh is preferred over install.sh, to prevent
# `make' implicit rules from creating a file called install from it
# when there is no Makefile.
#
# This script is compatible with the BSD install script, but was written
# from scratch.
#
# set DOITPROG to echo to test this script
# Don't use :- since 4.3BSD and earlier shells don't like it.
doit="${DOITPROG-}"
# put in absolute paths if you don't have them in your path; or use env. vars.
mvprog="${MVPROG-mv}"
cpprog="${CPPROG-cp}"
chmodprog="${CHMODPROG-chmod}"
chownprog="${CHOWNPROG-chown}"
chgrpprog="${CHGRPPROG-chgrp}"
stripprog="${STRIPPROG-strip}"
rmprog="${RMPROG-rm}"
mkdirprog="${MKDIRPROG-mkdir}"
transformbasename=""
transform_arg=""
instcmd="$mvprog"
chmodcmd="$chmodprog 0755"
chowncmd=""
chgrpcmd=""
stripcmd=""
rmcmd="$rmprog -f"
mvcmd="$mvprog"
src=""
dst=""
dir_arg=""
while [ x"$1" != x ]; do
case $1 in
-c) instcmd="$cpprog"
shift
continue;;
-d) dir_arg=true
shift
continue;;
-m) chmodcmd="$chmodprog $2"
shift
shift
continue;;
-o) chowncmd="$chownprog $2"
shift
shift
continue;;
-g) chgrpcmd="$chgrpprog $2"
shift
shift
continue;;
-s) stripcmd="$stripprog"
shift
continue;;
-t=*) transformarg=`echo $1 | sed 's/-t=//'`
shift
continue;;
-b=*) transformbasename=`echo $1 | sed 's/-b=//'`
shift
continue;;
*) if [ x"$src" = x ]
then
src=$1
else
# this colon is to work around a 386BSD /bin/sh bug
:
dst=$1
fi
shift
continue;;
esac
done
if [ x"$src" = x ]
then
echo "install: no input file specified"
exit 1
else
true
fi
if [ x"$dir_arg" != x ]; then
dst=$src
src=""
if [ -d $dst ]; then
instcmd=:
else
instcmd=mkdir
fi
else
# Waiting for this to be detected by the "$instcmd $src $dsttmp" command
# might cause directories to be created, which would be especially bad
# if $src (and thus $dsttmp) contains '*'.
if [ -f $src -o -d $src ]
then
true
else
echo "install: $src does not exist"
exit 1
fi
if [ x"$dst" = x ]
then
echo "install: no destination specified"
exit 1
else
true
fi
# If destination is a directory, append the input filename; if your system
# does not like double slashes in filenames, you may need to add some logic
if [ -d $dst ]
then
dst="$dst"/`basename $src`
else
true
fi
fi
## this sed command emulates the dirname command
dstdir=`echo $dst | sed -e 's,[^/]*$,,;s,/$,,;s,^$,.,'`
# Make sure that the destination directory exists.
# this part is taken from Noah Friedman's mkinstalldirs script
# Skip lots of stat calls in the usual case.
if [ ! -d "$dstdir" ]; then
defaultIFS='
'
IFS="${IFS-${defaultIFS}}"
oIFS="${IFS}"
# Some sh's can't handle IFS=/ for some reason.
IFS='%'
set - `echo ${dstdir} | sed -e 's@/@%@g' -e 's@^%@/@'`
IFS="${oIFS}"
pathcomp=''
while [ $# -ne 0 ] ; do
pathcomp="${pathcomp}${1}"
shift
if [ ! -d "${pathcomp}" ] ;
then
$mkdirprog "${pathcomp}"
else
true
fi
pathcomp="${pathcomp}/"
done
fi
if [ x"$dir_arg" != x ]
then
$doit $instcmd $dst &&
if [ x"$chowncmd" != x ]; then $doit $chowncmd $dst; else true ; fi &&
if [ x"$chgrpcmd" != x ]; then $doit $chgrpcmd $dst; else true ; fi &&
if [ x"$stripcmd" != x ]; then $doit $stripcmd $dst; else true ; fi &&
if [ x"$chmodcmd" != x ]; then $doit $chmodcmd $dst; else true ; fi
else
# If we're going to rename the final executable, determine the name now.
if [ x"$transformarg" = x ]
then
dstfile=`basename $dst`
else
dstfile=`basename $dst $transformbasename |
sed $transformarg`$transformbasename
fi
# don't allow the sed command to completely eliminate the filename
if [ x"$dstfile" = x ]
then
dstfile=`basename $dst`
else
true
fi
# Make a temp file name in the proper directory.
dsttmp=$dstdir/#inst.$$#
# Move or copy the file name to the temp name
$doit $instcmd $src $dsttmp &&
trap "rm -f ${dsttmp}" 0 &&
# and set any options; do chmod last to preserve setuid bits
# If any of these fail, we abort the whole thing. If we want to
# ignore errors from any of these, just make sure not to ignore
# errors from the above "$doit $instcmd $src $dsttmp" command.
if [ x"$chowncmd" != x ]; then $doit $chowncmd $dsttmp; else true;fi &&
if [ x"$chgrpcmd" != x ]; then $doit $chgrpcmd $dsttmp; else true;fi &&
if [ x"$stripcmd" != x ]; then $doit $stripcmd $dsttmp; else true;fi &&
if [ x"$chmodcmd" != x ]; then $doit $chmodcmd $dsttmp; else true;fi &&
# Now rename the file to the real destination.
$doit $rmcmd -f $dstdir/$dstfile &&
$doit $mvcmd $dsttmp $dstdir/$dstfile
fi &&
exit 0
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment