* Implemented the Terminal-ID search using a BClipboard and the new atomicity
feature of BClipboard::Commit(). * The file based version is now only used when the Terminal is built for BeOS. git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@20941 a95241bf-73f2-0310-859d-f6bbb57e9c96
This commit is contained in:
parent
f96bec013d
commit
d1c710a7d8
@ -17,6 +17,7 @@
|
|||||||
#include "TermConst.h"
|
#include "TermConst.h"
|
||||||
|
|
||||||
#include <Alert.h>
|
#include <Alert.h>
|
||||||
|
#include <Clipboard.h>
|
||||||
#include <NodeInfo.h>
|
#include <NodeInfo.h>
|
||||||
#include <Path.h>
|
#include <Path.h>
|
||||||
#include <Roster.h>
|
#include <Roster.h>
|
||||||
@ -428,6 +429,75 @@ TermApp::_IsMinimized(team_id id)
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*!
|
||||||
|
Checks if all teams that have an ID-to-team mapping in the message
|
||||||
|
are still running.
|
||||||
|
The IDs for teams that are gone will be made available again, and
|
||||||
|
their mapping is removed from the message.
|
||||||
|
*/
|
||||||
|
void
|
||||||
|
TermApp::_SanitizeIDs(BMessage* data, uint8* windows, ssize_t length)
|
||||||
|
{
|
||||||
|
BList teams;
|
||||||
|
be_roster->GetAppList(TERM_SIGNATURE, &teams);
|
||||||
|
|
||||||
|
for (int32 i = 0; i < length; i++) {
|
||||||
|
if (!windows[i])
|
||||||
|
continue;
|
||||||
|
|
||||||
|
BString id("id-");
|
||||||
|
id << i;
|
||||||
|
|
||||||
|
team_id team;
|
||||||
|
if (data->FindInt32(id.String(), &team) != B_OK)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
if (!teams.HasItem((void*)team)) {
|
||||||
|
windows[i] = false;
|
||||||
|
data->RemoveName(id.String());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*!
|
||||||
|
Removes the current fWindowNumber (ID) from the supplied array, or
|
||||||
|
finds a free ID in it, and sets fWindowNumber accordingly.
|
||||||
|
*/
|
||||||
|
bool
|
||||||
|
TermApp::_UpdateIDs(bool set, uint8* windows, ssize_t maxLength,
|
||||||
|
ssize_t* _length)
|
||||||
|
{
|
||||||
|
ssize_t length = *_length;
|
||||||
|
|
||||||
|
if (set) {
|
||||||
|
int32 i;
|
||||||
|
for (i = 0; i < length; i++) {
|
||||||
|
if (!windows[i]) {
|
||||||
|
windows[i] = true;
|
||||||
|
fWindowNumber = i + 1;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (i == length) {
|
||||||
|
if (length >= maxLength)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
windows[length] = true;
|
||||||
|
length++;
|
||||||
|
fWindowNumber = length;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// update information and write it back
|
||||||
|
windows[fWindowNumber - 1] = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
*_length = length;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
void
|
void
|
||||||
TermApp::_UpdateRegistration(bool set)
|
TermApp::_UpdateRegistration(bool set)
|
||||||
{
|
{
|
||||||
@ -436,6 +506,56 @@ TermApp::_UpdateRegistration(bool set)
|
|||||||
else if (fWindowNumber < 0)
|
else if (fWindowNumber < 0)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
|
#ifdef __HAIKU__
|
||||||
|
// use BClipboard - it supports atomic access in Haiku
|
||||||
|
BClipboard clipboard(TERM_SIGNATURE);
|
||||||
|
|
||||||
|
while (true) {
|
||||||
|
if (!clipboard.Lock())
|
||||||
|
return;
|
||||||
|
|
||||||
|
BMessage* data = clipboard.Data();
|
||||||
|
|
||||||
|
const uint8* windowsData;
|
||||||
|
uint8 windows[512];
|
||||||
|
ssize_t length;
|
||||||
|
if (data->FindData("ids", B_RAW_TYPE,
|
||||||
|
(const void**)&windowsData, &length) != B_OK)
|
||||||
|
length = 0;
|
||||||
|
|
||||||
|
if (length > (ssize_t)sizeof(windows))
|
||||||
|
length = sizeof(windows);
|
||||||
|
if (length > 0)
|
||||||
|
memcpy(windows, windowsData, length);
|
||||||
|
|
||||||
|
_SanitizeIDs(data, windows, length);
|
||||||
|
|
||||||
|
status_t status = B_OK;
|
||||||
|
if (_UpdateIDs(set, windows, sizeof(windows), &length)) {
|
||||||
|
// add/remove our ID-to-team mapping
|
||||||
|
BString id("id-");
|
||||||
|
id << fWindowNumber;
|
||||||
|
|
||||||
|
if (set)
|
||||||
|
data->AddInt32(id.String(), Team());
|
||||||
|
else
|
||||||
|
data->RemoveName(id.String());
|
||||||
|
|
||||||
|
data->RemoveName("ids");
|
||||||
|
//if (data->ReplaceData("ids", B_RAW_TYPE, windows, length) != B_OK)
|
||||||
|
data->AddData("ids", B_RAW_TYPE, windows, length);
|
||||||
|
|
||||||
|
status = clipboard.Commit(true);
|
||||||
|
}
|
||||||
|
|
||||||
|
clipboard.Unlock();
|
||||||
|
|
||||||
|
if (status == B_OK)
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
#else // !__HAIKU__
|
||||||
|
// use a file to store the IDs - unfortunately, locking
|
||||||
|
// doesn't work on BeOS either here
|
||||||
int fd = open("/tmp/terminal_ids", O_RDWR | O_CREAT);
|
int fd = open("/tmp/terminal_ids", O_RDWR | O_CREAT);
|
||||||
if (fd < 0)
|
if (fd < 0)
|
||||||
return;
|
return;
|
||||||
@ -447,41 +567,21 @@ TermApp::_UpdateRegistration(bool set)
|
|||||||
lock.l_len = -1;
|
lock.l_len = -1;
|
||||||
fcntl(fd, F_SETLKW, &lock);
|
fcntl(fd, F_SETLKW, &lock);
|
||||||
|
|
||||||
uint8 windows[256];
|
uint8 windows[512];
|
||||||
ssize_t length = read_pos(fd, 0, windows, sizeof(windows));
|
ssize_t length = read_pos(fd, 0, windows, sizeof(windows));
|
||||||
if (length < 0) {
|
if (length < 0) {
|
||||||
close(fd);
|
close(fd);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (length > (ssize_t)sizeof(windows))
|
if (length > (ssize_t)sizeof(windows))
|
||||||
length = sizeof(windows);
|
length = sizeof(windows);
|
||||||
|
|
||||||
if (set) {
|
if (_UpdateIDs(set, windows, sizeof(windows), &length))
|
||||||
int32 i;
|
|
||||||
for (i = 0; i < length; i++) {
|
|
||||||
if (!windows[i]) {
|
|
||||||
windows[i] = true;
|
|
||||||
fWindowNumber = i + 1;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (i == length) {
|
|
||||||
if (length >= (ssize_t)sizeof(windows)) {
|
|
||||||
close(fd);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
windows[length] = true;
|
|
||||||
length++;
|
|
||||||
fWindowNumber = length;
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
// update information and write it back
|
|
||||||
windows[fWindowNumber - 1] = false;
|
|
||||||
}
|
|
||||||
|
|
||||||
write_pos(fd, 0, windows, length);
|
write_pos(fd, 0, windows, length);
|
||||||
|
|
||||||
close(fd);
|
close(fd);
|
||||||
|
#endif // !__HAIKU__
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -62,6 +62,9 @@ class TermApp : public BApplication {
|
|||||||
void _SwitchTerm();
|
void _SwitchTerm();
|
||||||
void _ActivateTermWindow(team_id id);
|
void _ActivateTermWindow(team_id id);
|
||||||
bool _IsMinimized(team_id id);
|
bool _IsMinimized(team_id id);
|
||||||
|
void _SanitizeIDs(BMessage* data, uint8* windows, ssize_t length);
|
||||||
|
bool _UpdateIDs(bool set, uint8* windows, ssize_t maxLength,
|
||||||
|
ssize_t* _length);
|
||||||
void _UpdateRegistration(bool set);
|
void _UpdateRegistration(bool set);
|
||||||
void _UnregisterTerminal();
|
void _UnregisterTerminal();
|
||||||
void _RegisterTerminal();
|
void _RegisterTerminal();
|
||||||
|
Loading…
Reference in New Issue
Block a user