/** * @file demo_plugin.cpp * @brief Test plugin to be loaded by the llplugin testbed. * * $LicenseInfo:firstyear=2008&license=viewergpl$ * * Copyright (c) 2008-2009, Linden Research, Inc. * * Second Life Viewer Source Code * The source code in this file ("Source Code") is provided by Linden Lab * to you under the terms of the GNU General Public License, version 2.0 * ("GPL"), unless you have obtained a separate licensing agreement * ("Other License"), formally executed by you and Linden Lab. Terms of * the GPL can be found in doc/GPL-license.txt in this distribution, or * online at http://secondlifegrid.net/programs/open_source/licensing/gplv2 * * There are special exceptions to the terms and conditions of the GPL as * it is applied to this Source Code. View the full text of the exception * in the file doc/FLOSS-exception.txt in this software distribution, or * online at * http://secondlifegrid.net/programs/open_source/licensing/flossexception * * By copying, modifying or distributing this software, you acknowledge * that you have read and understood your obligations described above, * and agree to abide by those obligations. * * ALL LINDEN LAB SOURCE CODE IS PROVIDED "AS IS." LINDEN LAB MAKES NO * WARRANTIES, EXPRESS, IMPLIED OR OTHERWISE, REGARDING ITS ACCURACY, * COMPLETENESS OR PERFORMANCE. * $/LicenseInfo$ */ #include "linden_common.h" #include "llgl.h" #include "llplugininstance.h" #include "llpluginmessage.h" #include "llpluginmessageclasses.h" // TODO: Make sure that the only symbol exported from this library is LLPluginInitEntryPoint class DemoMediaPlugin2 { public: static int init(LLPluginInstance::sendMessageFunction host_send_func, void *host_user_data, LLPluginInstance::sendMessageFunction *plugin_send_func, void **plugin_user_data); private: DemoMediaPlugin2(LLPluginInstance::sendMessageFunction host_send_func, void *host_user_data); ~DemoMediaPlugin2(); static void staticReceiveMessage(const char *message_string, void **user_data); void receiveMessage(const char *message_string); void sendMessage(const LLPluginMessage &message); void setDirty(int left, int top, int right, int bottom); LLPluginInstance::sendMessageFunction mHostSendFunction; void *mHostUserData; bool mDeleteMe; class SharedSegmentInfo { public: void *mAddress; size_t mSize; }; typedef std::map SharedSegmentMap; SharedSegmentMap mSharedSegments; public: void updatePixels(void) { if ( mWidth < 1 || mWidth > 2048 || mHeight < 1 || mHeight > 2048 ) return; if ( mPixels == 0 ) return; if ( mFirstTime ) { // now we have a valid width/height - we can initialize positions for( int n = 0; n < NumObjects; ++n ) { mXpos[ n ] = ( mWidth / 2 ) + rand() % ( mWidth / 16 ) - ( mWidth / 32 ); mYpos[ n ] = ( mHeight / 2 ) + rand() % ( mHeight / 16 ) - ( mHeight / 32 ); mColorR[ n ] = rand() % 0xa0 + 0x60; mColorG[ n ] = rand() % 0xa0 + 0x60; mColorB[ n ] = rand() % 0xa0 + 0x60; mXInc[ n ] = 0; while ( mXInc[ n ] == 0 ) mXInc[ n ] = rand() % 7 - 3; mYInc[ n ] = 0; while ( mYInc[ n ] == 0 ) mYInc[ n ] = rand() % 9 - 4; mBlockSize[ n ] = rand() % 0x16 + 0x04; }; delete [] mBackgroundPixels; mBackgroundPixels = new unsigned char[ mWidth * mHeight * mDepth ]; mFirstTime = false; }; if ( time( NULL ) > mLastUpdateTime + 3 ) { const int num_squares = rand() % 20 + 4; int sqr1_r = rand() % 0x80 + 0x20; int sqr1_g = rand() % 0x80 + 0x20; int sqr1_b = rand() % 0x80 + 0x20; int sqr2_r = rand() % 0x80 + 0x20; int sqr2_g = rand() % 0x80 + 0x20; int sqr2_b = rand() % 0x80 + 0x20; for ( int y1 = 0; y1 < num_squares; ++y1 ) { for ( int x1 = 0; x1 < num_squares; ++x1 ) { int px_start = mWidth * x1 / num_squares; int px_end = ( mWidth * ( x1 + 1 ) ) / num_squares; int py_start = mHeight * y1 / num_squares; int py_end = ( mHeight * ( y1 + 1 ) ) / num_squares; for( int y2 = py_start; y2 < py_end; ++y2 ) { for( int x2 = px_start; x2 < px_end; ++x2 ) { int rowspan = mWidth * mDepth; if ( ( y1 % 2 ) ^ ( x1 % 2 ) ) { mBackgroundPixels[ y2 * rowspan + x2 * mDepth + 0 ] = sqr1_r; mBackgroundPixels[ y2 * rowspan + x2 * mDepth + 1 ] = sqr1_g; mBackgroundPixels[ y2 * rowspan + x2 * mDepth + 2 ] = sqr1_b; } else { mBackgroundPixels[ y2 * rowspan + x2 * mDepth + 0 ] = sqr2_r; mBackgroundPixels[ y2 * rowspan + x2 * mDepth + 1 ] = sqr2_g; mBackgroundPixels[ y2 * rowspan + x2 * mDepth + 2 ] = sqr2_b; }; }; }; }; }; time( &mLastUpdateTime ); }; memcpy( mPixels, mBackgroundPixels, mWidth * mHeight * mDepth ); for( int n = 0; n < NumObjects; ++n ) { if ( rand() % 50 == 0 ) { mXInc[ n ] = 0; while ( mXInc[ n ] == 0 ) mXInc[ n ] = rand() % 7 - 3; mYInc[ n ] = 0; while ( mYInc[ n ] == 0 ) mYInc[ n ] = rand() % 9 - 4; }; if ( mXpos[ n ] + mXInc[ n ] < 0 || mXpos[ n ] + mXInc[ n ] >= mWidth - mBlockSize[ n ] ) mXInc[ n ] =- mXInc[ n ]; if ( mYpos[ n ] + mYInc[ n ] < 0 || mYpos[ n ] + mYInc[ n ] >= mHeight - mBlockSize[ n ] ) mYInc[ n ] =- mYInc[ n ]; mXpos[ n ] += mXInc[ n ]; mYpos[ n ] += mYInc[ n ]; for( int y = 0; y < mBlockSize[ n ]; ++y ) { for( int x = 0; x < mBlockSize[ n ]; ++x ) { mPixels[ ( mXpos[ n ] + x ) * mDepth + ( mYpos[ n ] + y ) * mDepth * mWidth + 0 ] = mColorR[ n ]; mPixels[ ( mXpos[ n ] + x ) * mDepth + ( mYpos[ n ] + y ) * mDepth * mWidth + 1 ] = mColorG[ n ]; mPixels[ ( mXpos[ n ] + x ) * mDepth + ( mYpos[ n ] + y ) * mDepth * mWidth + 2 ] = mColorB[ n ]; }; }; }; // TODO: experiment with setDirty to just follow block position setDirty( 0, 0, mWidth, mHeight ); }; //////////////////////////////////////////////////////////////////////////////// // void update() { updatePixels(); }; //////////////////////////////////////////////////////////////////////////////// // void write_pixel( int x, int y, unsigned char r, unsigned char g, unsigned char b ) { // make sure we don't write outside the buffer if((x < 0) || (x >= mWidth) || (y < 0) || (y >= mHeight)) return; if( mBackgroundPixels != NULL) { unsigned char *pixel = mBackgroundPixels; pixel += y * mWidth * mDepth; // row offset pixel += (x * mDepth); // columm offset pixel[0] = b; pixel[1] = g; pixel[2] = r; setDirty(x, y, x+1, y+1); } } //////////////////////////////////////////////////////////////////////////////// // void mouseDown( int x, int y ) { write_pixel( x, y, 0xff, 0x00, 0x00 ); }; //////////////////////////////////////////////////////////////////////////////// // void mouseUp( int x, int y ) { write_pixel( x, y, 0xff, 0xff, 0x00 ); }; //////////////////////////////////////////////////////////////////////////////// // void mouseMove( int x, int y ) { write_pixel( x , y , 0xff, 0x00, 0xff ); }; //////////////////////////////////////////////////////////////////////////////// // void keyPress( unsigned char key ) { }; private: unsigned char* mPixels; int mWidth; int mHeight; int mTextureWidth; int mTextureHeight; int mDepth; time_t mLastUpdateTime; bool mFirstTime; unsigned char* mBackgroundPixels; enum Constants2 { NumObjects = 20 }; int mColorR[ NumObjects ]; int mColorG[ NumObjects ]; int mColorB[ NumObjects ]; int mXpos[ NumObjects ]; int mYpos[ NumObjects ]; int mXInc[ NumObjects ]; int mYInc[ NumObjects ]; int mBlockSize[ NumObjects ]; }; int DemoMediaPlugin2::init(LLPluginInstance::sendMessageFunction host_send_func, void *host_user_data, LLPluginInstance::sendMessageFunction *plugin_send_func, void **plugin_user_data) { DemoMediaPlugin2 *self = new DemoMediaPlugin2(host_send_func, host_user_data); *plugin_send_func = staticReceiveMessage; *plugin_user_data = (void*)self; return 0; } DemoMediaPlugin2::DemoMediaPlugin2(LLPluginInstance::sendMessageFunction host_send_func, void *host_user_data) { std::cerr << "DemoMediaPlugin2 constructor" << std::endl; mHostSendFunction = host_send_func; mHostUserData = host_user_data; mDeleteMe = false; mLastUpdateTime = 0; mFirstTime = true; mLastUpdateTime = 0; mWidth = 0; mHeight = 0; mDepth = 0; mPixels = 0; mBackgroundPixels = 0; srand( get_clock_count() ) ; } DemoMediaPlugin2::~DemoMediaPlugin2() { std::cerr << "DemoMediaPlugin2 destructor" << std::endl; } void DemoMediaPlugin2::staticReceiveMessage(const char *message_string, void **user_data) { DemoMediaPlugin2 *self = (DemoMediaPlugin2*)*user_data; if(self != NULL) { self->receiveMessage(message_string); // If the plugin has processed the delete message, delete it. if(self->mDeleteMe) { delete self; *user_data = NULL; } } } void DemoMediaPlugin2::receiveMessage(const char *message_string) { // std::cerr << "DemoMediaPlugin2::receiveMessage: received message: \"" << message_string << "\"" << std::endl; LLPluginMessage message_in; if(message_in.parse(message_string) >= 0) { std::string message_class = message_in.getClass(); std::string message_name = message_in.getName(); if(message_class == LLPLUGIN_MESSAGE_CLASS_BASE) { if(message_name == "init") { LLPluginMessage message("base", "init_response"); LLSD versions = LLSD::emptyMap(); versions[LLPLUGIN_MESSAGE_CLASS_BASE] = LLPLUGIN_MESSAGE_CLASS_BASE_VERSION; versions[LLPLUGIN_MESSAGE_CLASS_MEDIA] = LLPLUGIN_MESSAGE_CLASS_MEDIA_VERSION; // Normally a plugin would only specify one of these two subclasses, but this is a demo... versions[LLPLUGIN_MESSAGE_CLASS_MEDIA_BROWSER] = LLPLUGIN_MESSAGE_CLASS_MEDIA_BROWSER_VERSION; versions[LLPLUGIN_MESSAGE_CLASS_MEDIA_TIME] = LLPLUGIN_MESSAGE_CLASS_MEDIA_TIME_VERSION; message.setValueLLSD("versions", versions); sendMessage(message); // Plugin gets to decide the texture parameters to use. mDepth = 3; message.setMessage(LLPLUGIN_MESSAGE_CLASS_MEDIA, "texture_params"); message.setValueS32("depth", mDepth); message.setValueU32("internalformat", GL_RGB); message.setValueU32("format", GL_RGB); message.setValueU32("type", GL_UNSIGNED_BYTE); message.setValueBoolean("coords_opengl", false); // true == use OpenGL-style coordinates, false == (0,0) is upper left. sendMessage(message); } else if(message_name == "idle") { // no response is necessary here. update(); } else if(message_name == "shutdown") { sendMessage(LLPluginMessage("base", "shutdown_response")); mDeleteMe = true; } else if(message_name == "shm_added") { SharedSegmentInfo info; info.mAddress = (void*)message_in.getValueU32("address"); info.mSize = (size_t)message_in.getValueS32("size"); std::string name = message_in.getValue("name"); std::cerr << "DemoMediaPlugin2::receiveMessage: shared memory added, name: " << name << ", size: " << info.mSize << ", address: " << info.mAddress << std::endl; mSharedSegments.insert(SharedSegmentMap::value_type(name, info)); } else if(message_name == "shm_remove") { std::string name = message_in.getValue("name"); std::cerr << "DemoMediaPlugin2::receiveMessage: shared memory remove, name = " << name << std::endl; SharedSegmentMap::iterator iter = mSharedSegments.find(name); if(iter != mSharedSegments.end()) { if(mPixels == iter->second.mAddress) { // This is the currently active pixel buffer. Make sure we stop drawing to it. mPixels = NULL; } mSharedSegments.erase(iter); } else { std::cerr << "DemoMediaPlugin2::receiveMessage: unknown shared memory region!" << std::endl; } // Send the response so it can be cleaned up. LLPluginMessage message("base", "shm_remove_response"); message.setValue("name", name); sendMessage(message); } else { std::cerr << "DemoMediaPlugin2::receiveMessage: unknown base message: " << message_name << std::endl; } } else if(message_class == LLPLUGIN_MESSAGE_CLASS_MEDIA) { if(message_name == "size_change") { std::string name = message_in.getValue("name"); S32 width = message_in.getValueS32("width"); S32 height = message_in.getValueS32("height"); S32 texture_width = message_in.getValueS32("texture_width"); S32 texture_height = message_in.getValueS32("texture_height"); LLPluginMessage message(LLPLUGIN_MESSAGE_CLASS_MEDIA, "size_change_response"); message.setValue("name", name); message.setValueS32("width", width); message.setValueS32("height", height); message.setValueS32("texture_width", texture_width); message.setValueS32("texture_height", texture_height); sendMessage(message); if(!name.empty()) { // Find the shared memory region with this name SharedSegmentMap::iterator iter = mSharedSegments.find(name); if(iter != mSharedSegments.end()) { std::cerr << "Got size change, new size is " << width << " by " << height << std::endl; std::cerr << " texture size is " << texture_width << " by " << texture_height << std::endl; mPixels = (unsigned char*)iter->second.mAddress; mWidth = width; mHeight = height; mTextureWidth = texture_width; mTextureHeight = texture_height; mFirstTime = true; updatePixels(); } } } else if(message_name == "mouse_event") { std::string event = message_in.getValue("event"); S32 x = message_in.getValueS32("x"); S32 y = message_in.getValueS32("y"); // std::string modifiers = message.getValue("modifiers"); // std::cerr << "DemoMediaPlugin2::receiveMessage: mouse event \"" << event // << "\", coords " << x << ", " << y // << std::endl; if(event == "down") { mouseDown(x, y); } else if(event == "up") { mouseUp(x, y); } else if(event == "move") { mouseMove(x, y); } } else { std::cerr << "DemoMediaPlugin2::receiveMessage: unknown media message: " << message_string << std::endl; } } else if(message_class == LLPLUGIN_MESSAGE_CLASS_MEDIA_BROWSER) { if(message_name == "focus") { // foo = message_in.getValueBoolean("focused"); } else if(message_name == "clear_cache") { } else if(message_name == "clear_cookies") { } else if(message_name == "enable_cookies") { // foo = message_in.getValueBoolean("enable"); } else if(message_name == "proxy_setup") { // foo = message_in.getValueBoolean("enable"); // bar = message_in.getValue("host"); // baz = message_in.getValueS32("port"); } else if(message_name == "browse_stop") { } else if(message_name == "browse_reload") { // foo = message_in.getValueBoolean("ignore_cache"); } else if(message_name == "browse_forward") { } else if(message_name == "browse_back") { } else if(message_name == "set_status_redirect") { // foo = message_in.getValueS32("code"); // bar = message_in.getValue("url"); } else { std::cerr << "DemoMediaPlugin2::receiveMessage: unknown media_browser message: " << message_string << std::endl; } } else { std::cerr << "DemoMediaPlugin2::receiveMessage: unknown message class: " << message_class << std::endl; } } } void DemoMediaPlugin2::sendMessage(const LLPluginMessage &message) { std::string output = message.generate(); mHostSendFunction(output.c_str(), &mHostUserData); } void DemoMediaPlugin2::setDirty(int left, int top, int right, int bottom) { LLPluginMessage message(LLPLUGIN_MESSAGE_CLASS_MEDIA, "updated"); message.setValueS32("left", left); message.setValueS32("top", top); message.setValueS32("right", right); message.setValueS32("bottom", bottom); sendMessage(message); } extern "C" { #ifdef WIN32 __declspec(dllexport) #endif int LLPluginInitEntryPoint(LLPluginInstance::sendMessageFunction host_send_func, void *host_user_data, LLPluginInstance::sendMessageFunction *plugin_send_func, void **plugin_user_data); } int #ifdef WIN32 __declspec(dllexport) #endif LLPluginInitEntryPoint(LLPluginInstance::sendMessageFunction host_send_func, void *host_user_data, LLPluginInstance::sendMessageFunction *plugin_send_func, void **plugin_user_data) { return DemoMediaPlugin2::init(host_send_func, host_user_data, plugin_send_func, plugin_user_data); } #ifdef WIN32 int WINAPI DllEntryPoint( HINSTANCE hInstance, unsigned long reason, void* params ) { return 1; } #endif