2014-10-23 23:23:26 +02:00
|
|
|
/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
|
|
|
|
/*
|
|
|
|
* This file is part of the LibreOffice project.
|
|
|
|
*
|
|
|
|
* This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/.
|
|
|
|
*
|
|
|
|
* This file incorporates work covered by the following license notice:
|
|
|
|
*
|
|
|
|
* Licensed to the Apache Software Foundation (ASF) under one or more
|
|
|
|
* contributor license agreements. See the NOTICE file distributed
|
|
|
|
* with this work for additional information regarding copyright
|
|
|
|
* ownership. The ASF licenses this file to you under the Apache
|
|
|
|
* License, Version 2.0 (the "License"); you may not use this file
|
|
|
|
* except in compliance with the License. You may obtain a copy of
|
|
|
|
* the License at http://www.apache.org/licenses/LICENSE-2.0 .
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef INCLUDED_VCL_OPENGLGDIIMPL_HXX
|
|
|
|
#define INCLUDED_VCL_OPENGLGDIIMPL_HXX
|
|
|
|
|
2014-11-29 22:55:31 +00:00
|
|
|
#include "salgeom.hxx"
|
2014-10-23 23:23:26 +02:00
|
|
|
#include "salgdiimpl.hxx"
|
2014-10-23 23:32:04 +02:00
|
|
|
#include <vcl/dllapi.h>
|
2014-10-23 23:23:26 +02:00
|
|
|
|
2014-11-26 09:22:25 -05:00
|
|
|
#include "opengl/framebuffer.hxx"
|
2014-11-28 14:56:08 -05:00
|
|
|
#include "opengl/program.hxx"
|
2014-11-10 13:46:12 -05:00
|
|
|
#include "opengl/texture.hxx"
|
2014-11-22 08:09:29 -05:00
|
|
|
#include "regionband.hxx"
|
2014-11-10 13:46:12 -05:00
|
|
|
|
2014-11-11 05:13:40 -05:00
|
|
|
#include <tools/poly.hxx>
|
2014-10-24 18:36:31 +02:00
|
|
|
#include <vcl/opengl/OpenGLContext.hxx>
|
|
|
|
|
2014-11-03 11:08:24 -05:00
|
|
|
class SalFrame;
|
2014-11-08 19:23:16 +01:00
|
|
|
class SalVirtualDevice;
|
2015-08-29 23:15:54 +01:00
|
|
|
class OpenGLTests;
|
2014-11-03 11:08:24 -05:00
|
|
|
|
2015-01-18 19:00:32 +01:00
|
|
|
namespace basegfx
|
|
|
|
{
|
|
|
|
class B2DTrapezoid;
|
|
|
|
};
|
|
|
|
|
2015-08-03 15:06:55 +09:00
|
|
|
struct TextureCombo
|
|
|
|
{
|
|
|
|
std::unique_ptr<OpenGLTexture> mpTexture;
|
|
|
|
std::unique_ptr<OpenGLTexture> mpMask;
|
|
|
|
};
|
|
|
|
|
2015-08-29 23:15:54 +01:00
|
|
|
class VCL_DLLPUBLIC OpenGLSalGraphicsImpl : public SalGraphicsImpl
|
2014-10-23 23:23:26 +02:00
|
|
|
{
|
2015-08-29 23:15:54 +01:00
|
|
|
friend class OpenGLTests;
|
2014-11-08 13:14:14 -05:00
|
|
|
protected:
|
2014-10-24 18:36:31 +02:00
|
|
|
|
2014-12-02 21:51:50 +01:00
|
|
|
OpenGLContext* mpContext;
|
2014-12-15 20:00:45 +01:00
|
|
|
SalGraphics& mrParent;
|
2014-11-29 22:55:31 +00:00
|
|
|
/// Pointer to the SalFrame or SalVirtualDevice
|
2014-12-15 20:00:45 +01:00
|
|
|
SalGeometryProvider* mpProvider;
|
2014-11-26 09:22:25 -05:00
|
|
|
OpenGLFramebuffer* mpFramebuffer;
|
2014-11-28 14:56:08 -05:00
|
|
|
OpenGLProgram* mpProgram;
|
2014-10-24 18:36:31 +02:00
|
|
|
|
2014-11-12 15:37:11 -05:00
|
|
|
// clipping
|
2014-11-22 08:07:47 -05:00
|
|
|
vcl::Region maClipRegion;
|
2014-11-12 15:37:11 -05:00
|
|
|
bool mbUseScissor;
|
|
|
|
bool mbUseStencil;
|
|
|
|
|
2014-11-10 13:46:12 -05:00
|
|
|
bool mbOffscreen;
|
2014-11-13 21:37:54 -05:00
|
|
|
OpenGLTexture maOffscreenTex;
|
2014-11-10 13:46:12 -05:00
|
|
|
|
2014-10-29 13:05:02 -04:00
|
|
|
SalColor mnLineColor;
|
|
|
|
SalColor mnFillColor;
|
2014-12-18 11:47:58 +01:00
|
|
|
#ifdef DBG_UTIL
|
2015-01-12 13:20:54 +01:00
|
|
|
bool mProgramIsSolidColor;
|
2014-12-18 11:47:58 +01:00
|
|
|
#endif
|
2015-01-12 13:20:54 +01:00
|
|
|
SalColor mProgramSolidColor;
|
|
|
|
double mProgramSolidTransparency;
|
2014-10-29 13:05:02 -04:00
|
|
|
|
2014-11-22 08:07:47 -05:00
|
|
|
void ImplInitClipRegion();
|
2014-11-12 18:11:34 -05:00
|
|
|
void ImplSetClipBit( const vcl::Region& rClip, GLuint nMask );
|
2015-01-12 13:24:47 +01:00
|
|
|
void ImplDrawLineAA( double nX1, double nY1, double nX2, double nY2, bool edge = false );
|
2014-11-13 21:38:58 -05:00
|
|
|
bool CheckOffscreenTexture();
|
|
|
|
|
2015-07-08 18:43:32 +09:00
|
|
|
void ApplyProgramMatrices(float fPixelOffset = 0.0);
|
|
|
|
|
2014-11-13 22:24:35 -05:00
|
|
|
public:
|
2015-01-20 14:48:48 +01:00
|
|
|
bool UseProgram( const OUString& rVertexShader, const OUString& rFragmentShader, const OString& preamble = "" );
|
2014-11-28 14:56:08 -05:00
|
|
|
bool UseSolid( SalColor nColor, sal_uInt8 nTransparency );
|
|
|
|
bool UseSolid( SalColor nColor, double fTransparency );
|
|
|
|
bool UseSolid( SalColor nColor );
|
2014-12-18 11:47:58 +01:00
|
|
|
bool UseSolidAA( SalColor nColor, double fTransparency );
|
2014-12-15 20:00:45 +01:00
|
|
|
bool UseSolidAA( SalColor nColor );
|
2014-11-28 14:56:08 -05:00
|
|
|
bool UseInvert();
|
2014-10-29 13:05:02 -04:00
|
|
|
|
|
|
|
void DrawPoint( long nX, long nY );
|
2014-12-18 11:47:58 +01:00
|
|
|
void DrawLine( double nX1, double nY1, double nX2, double nY2 );
|
|
|
|
void DrawLineAA( double nX1, double nY1, double nX2, double nY2 );
|
2014-12-15 20:00:45 +01:00
|
|
|
void DrawLinesAA( sal_uInt32 nPoints, const SalPoint* pPtAry, bool bClose );
|
2015-01-12 13:24:47 +01:00
|
|
|
void DrawEdgeAA( double nX1, double nY1, double nX2, double nY2 );
|
2015-01-18 18:58:20 +01:00
|
|
|
void DrawConvexPolygon( sal_uInt32 nPoints, const SalPoint* pPtAry, bool blockAA = false );
|
2015-08-16 16:45:12 -05:00
|
|
|
void DrawConvexPolygon( const tools::Polygon& rPolygon, bool blockAA = false );
|
2015-01-18 22:42:06 +01:00
|
|
|
void DrawTrapezoid( const basegfx::B2DTrapezoid& trapezoid, bool blockAA = false );
|
2014-10-30 21:51:39 -04:00
|
|
|
void DrawRect( long nX, long nY, long nWidth, long nHeight );
|
2014-11-11 15:54:03 -05:00
|
|
|
void DrawRect( const Rectangle& rRect );
|
2014-10-29 13:05:02 -04:00
|
|
|
void DrawPolygon( sal_uInt32 nPoints, const SalPoint* pPtAry );
|
2015-01-12 13:20:54 +01:00
|
|
|
void DrawPolyPolygon( const basegfx::B2DPolyPolygon& rPolyPolygon, bool blockAA = false );
|
2014-11-22 08:09:29 -05:00
|
|
|
void DrawRegionBand( const RegionBand& rRegion );
|
2014-11-13 21:37:54 -05:00
|
|
|
void DrawTextureRect( OpenGLTexture& rTexture, const SalTwoRect& rPosAry, bool bInverted = false );
|
|
|
|
void DrawTexture( OpenGLTexture& rTexture, const SalTwoRect& rPosAry, bool bInverted = false );
|
2014-11-17 09:15:15 -05:00
|
|
|
void DrawTransformedTexture( OpenGLTexture& rTexture, OpenGLTexture& rMask, const basegfx::B2DPoint& rNull, const basegfx::B2DPoint& rX, const basegfx::B2DPoint& rY );
|
2014-11-13 22:24:35 -05:00
|
|
|
void DrawAlphaTexture( OpenGLTexture& rTexture, const SalTwoRect& rPosAry, bool bInverted = false, bool pPremultiplied = false );
|
2014-11-20 22:07:12 -05:00
|
|
|
void DrawTextureDiff( OpenGLTexture& rTexture, OpenGLTexture& rMask, const SalTwoRect& rPosAry, bool bInverted = false );
|
2014-11-13 21:37:54 -05:00
|
|
|
void DrawTextureWithMask( OpenGLTexture& rTexture, OpenGLTexture& rMask, const SalTwoRect& rPosAry );
|
2014-11-18 12:34:53 -05:00
|
|
|
void DrawBlendedTexture( OpenGLTexture& rTexture, OpenGLTexture& rMask, OpenGLTexture& rAlpha, const SalTwoRect& rPosAry );
|
2014-11-13 21:37:54 -05:00
|
|
|
void DrawMask( OpenGLTexture& rTexture, SalColor nMaskColor, const SalTwoRect& rPosAry );
|
2014-11-11 05:13:40 -05:00
|
|
|
void DrawLinearGradient( const Gradient& rGradient, const Rectangle& rRect );
|
2014-11-14 01:16:28 -05:00
|
|
|
void DrawAxialGradient( const Gradient& rGradient, const Rectangle& rRect );
|
2014-11-11 15:54:03 -05:00
|
|
|
void DrawRadialGradient( const Gradient& rGradient, const Rectangle& rRect );
|
2014-10-29 13:05:02 -04:00
|
|
|
|
2014-11-13 22:24:35 -05:00
|
|
|
public:
|
2014-11-08 13:14:14 -05:00
|
|
|
// get the width of the device
|
2014-12-15 20:00:45 +01:00
|
|
|
GLfloat GetWidth() const { return mpProvider ? mpProvider->GetWidth() : 1; }
|
2014-11-08 13:14:14 -05:00
|
|
|
|
|
|
|
// get the height of the device
|
2014-12-15 20:00:45 +01:00
|
|
|
GLfloat GetHeight() const { return mpProvider ? mpProvider->GetHeight() : 1; }
|
2014-11-08 13:14:14 -05:00
|
|
|
|
2014-11-22 07:58:38 -05:00
|
|
|
// check whether this instance is used for offscreen rendering
|
2015-04-24 12:34:53 +02:00
|
|
|
bool IsOffscreen() const { return mpProvider == nullptr || mpProvider->IsOffScreen(); }
|
2014-11-22 07:58:38 -05:00
|
|
|
|
2014-11-10 13:06:38 -05:00
|
|
|
// operations to do before painting
|
2015-01-20 12:38:10 +02:00
|
|
|
void PreDraw();
|
2014-11-10 13:06:38 -05:00
|
|
|
|
|
|
|
// operations to do after painting
|
2015-01-20 12:38:10 +02:00
|
|
|
void PostDraw();
|
2014-11-10 13:06:38 -05:00
|
|
|
|
2014-11-22 07:58:38 -05:00
|
|
|
protected:
|
2014-11-26 09:22:25 -05:00
|
|
|
bool AcquireContext();
|
2014-11-22 07:58:38 -05:00
|
|
|
bool ReleaseContext();
|
|
|
|
|
2014-12-02 09:05:19 -05:00
|
|
|
// retrieve the default context for offscreen rendering
|
2015-04-01 08:33:09 +02:00
|
|
|
static OpenGLContext* GetDefaultContext();
|
2014-12-02 09:05:19 -05:00
|
|
|
|
2014-11-22 07:58:38 -05:00
|
|
|
// create a new context for window rendering
|
|
|
|
virtual OpenGLContext* CreateWinContext() = 0;
|
2014-11-10 13:46:12 -05:00
|
|
|
|
2014-11-26 09:22:25 -05:00
|
|
|
// check whether the given context can be used by this instance
|
|
|
|
virtual bool UseContext( OpenGLContext* pContext ) = 0;
|
|
|
|
|
2014-10-23 23:23:26 +02:00
|
|
|
public:
|
2014-12-15 20:00:45 +01:00
|
|
|
OpenGLSalGraphicsImpl(SalGraphics& pParent, SalGeometryProvider *pProvider);
|
2014-10-23 23:23:26 +02:00
|
|
|
virtual ~OpenGLSalGraphicsImpl ();
|
|
|
|
|
2014-11-26 09:22:25 -05:00
|
|
|
OpenGLContext* GetOpenGLContext();
|
2014-11-22 07:58:38 -05:00
|
|
|
|
|
|
|
virtual void Init() SAL_OVERRIDE;
|
2014-10-25 02:16:51 +02:00
|
|
|
|
2015-09-01 15:09:25 +01:00
|
|
|
virtual void DeInit() SAL_OVERRIDE;
|
|
|
|
|
2014-10-23 23:23:26 +02:00
|
|
|
virtual void freeResources() SAL_OVERRIDE;
|
|
|
|
|
2015-01-20 12:38:10 +02:00
|
|
|
const vcl::Region& getClipRegion() const;
|
2014-10-23 23:23:26 +02:00
|
|
|
virtual bool setClipRegion( const vcl::Region& ) SAL_OVERRIDE;
|
2014-12-04 11:45:55 -05:00
|
|
|
|
2014-10-23 23:23:26 +02:00
|
|
|
//
|
|
|
|
// get the depth of the device
|
|
|
|
virtual sal_uInt16 GetBitCount() const SAL_OVERRIDE;
|
|
|
|
|
|
|
|
// get the width of the device
|
|
|
|
virtual long GetGraphicsWidth() const SAL_OVERRIDE;
|
|
|
|
|
|
|
|
// set the clip region to empty
|
|
|
|
virtual void ResetClipRegion() SAL_OVERRIDE;
|
|
|
|
|
|
|
|
// set the line color to transparent (= don't draw lines)
|
|
|
|
|
|
|
|
virtual void SetLineColor() SAL_OVERRIDE;
|
|
|
|
|
|
|
|
// set the line color to a specific color
|
|
|
|
virtual void SetLineColor( SalColor nSalColor ) SAL_OVERRIDE;
|
|
|
|
|
|
|
|
// set the fill color to transparent (= don't fill)
|
|
|
|
virtual void SetFillColor() SAL_OVERRIDE;
|
|
|
|
|
|
|
|
// set the fill color to a specific color, shapes will be
|
|
|
|
// filled accordingly
|
|
|
|
virtual void SetFillColor( SalColor nSalColor ) SAL_OVERRIDE;
|
|
|
|
|
|
|
|
// enable/disable XOR drawing
|
|
|
|
virtual void SetXORMode( bool bSet, bool bInvertOnly ) SAL_OVERRIDE;
|
|
|
|
|
|
|
|
// set line color for raster operations
|
|
|
|
virtual void SetROPLineColor( SalROPColor nROPColor ) SAL_OVERRIDE;
|
|
|
|
|
|
|
|
// set fill color for raster operations
|
|
|
|
virtual void SetROPFillColor( SalROPColor nROPColor ) SAL_OVERRIDE;
|
|
|
|
|
|
|
|
// draw --> LineColor and FillColor and RasterOp and ClipRegion
|
|
|
|
virtual void drawPixel( long nX, long nY ) SAL_OVERRIDE;
|
|
|
|
virtual void drawPixel( long nX, long nY, SalColor nSalColor ) SAL_OVERRIDE;
|
|
|
|
|
|
|
|
virtual void drawLine( long nX1, long nY1, long nX2, long nY2 ) SAL_OVERRIDE;
|
|
|
|
|
|
|
|
virtual void drawRect( long nX, long nY, long nWidth, long nHeight ) SAL_OVERRIDE;
|
|
|
|
|
|
|
|
virtual void drawPolyLine( sal_uInt32 nPoints, const SalPoint* pPtAry ) SAL_OVERRIDE;
|
|
|
|
|
|
|
|
virtual void drawPolygon( sal_uInt32 nPoints, const SalPoint* pPtAry ) SAL_OVERRIDE;
|
|
|
|
|
|
|
|
virtual void drawPolyPolygon( sal_uInt32 nPoly, const sal_uInt32* pPoints, PCONSTSALPOINT* pPtAry ) SAL_OVERRIDE;
|
|
|
|
virtual bool drawPolyPolygon( const ::basegfx::B2DPolyPolygon&, double fTransparency ) SAL_OVERRIDE;
|
|
|
|
|
|
|
|
virtual bool drawPolyLine(
|
|
|
|
const ::basegfx::B2DPolygon&,
|
|
|
|
double fTransparency,
|
|
|
|
const ::basegfx::B2DVector& rLineWidths,
|
|
|
|
basegfx::B2DLineJoin,
|
|
|
|
com::sun::star::drawing::LineCap) SAL_OVERRIDE;
|
|
|
|
|
|
|
|
virtual bool drawPolyLineBezier(
|
|
|
|
sal_uInt32 nPoints,
|
|
|
|
const SalPoint* pPtAry,
|
|
|
|
const sal_uInt8* pFlgAry ) SAL_OVERRIDE;
|
|
|
|
|
|
|
|
virtual bool drawPolygonBezier(
|
|
|
|
sal_uInt32 nPoints,
|
|
|
|
const SalPoint* pPtAry,
|
|
|
|
const sal_uInt8* pFlgAry ) SAL_OVERRIDE;
|
|
|
|
|
|
|
|
virtual bool drawPolyPolygonBezier(
|
|
|
|
sal_uInt32 nPoly,
|
|
|
|
const sal_uInt32* pPoints,
|
|
|
|
const SalPoint* const* pPtAry,
|
|
|
|
const sal_uInt8* const* pFlgAry ) SAL_OVERRIDE;
|
|
|
|
|
|
|
|
// CopyArea --> No RasterOp, but ClipRegion
|
|
|
|
virtual void copyArea(
|
|
|
|
long nDestX, long nDestY,
|
|
|
|
long nSrcX, long nSrcY,
|
|
|
|
long nSrcWidth, long nSrcHeight,
|
|
|
|
sal_uInt16 nFlags ) SAL_OVERRIDE;
|
|
|
|
|
|
|
|
// CopyBits and DrawBitmap --> RasterOp and ClipRegion
|
|
|
|
// CopyBits() --> pSrcGraphics == NULL, then CopyBits on same Graphics
|
2014-11-19 09:58:05 +00:00
|
|
|
void DoCopyBits(const SalTwoRect& rPosAry, OpenGLSalGraphicsImpl &rSrcImpl);
|
2014-10-23 23:23:26 +02:00
|
|
|
|
2014-11-18 12:34:53 -05:00
|
|
|
virtual bool blendBitmap(
|
|
|
|
const SalTwoRect&,
|
|
|
|
const SalBitmap& rBitmap ) SAL_OVERRIDE;
|
|
|
|
|
|
|
|
virtual bool blendAlphaBitmap(
|
|
|
|
const SalTwoRect&,
|
|
|
|
const SalBitmap& rSrcBitmap,
|
|
|
|
const SalBitmap& rMaskBitmap,
|
|
|
|
const SalBitmap& rAlphaBitmap ) SAL_OVERRIDE;
|
|
|
|
|
2014-10-23 23:23:26 +02:00
|
|
|
virtual void drawBitmap( const SalTwoRect& rPosAry, const SalBitmap& rSalBitmap ) SAL_OVERRIDE;
|
|
|
|
|
|
|
|
virtual void drawBitmap(
|
|
|
|
const SalTwoRect& rPosAry,
|
|
|
|
const SalBitmap& rSalBitmap,
|
|
|
|
const SalBitmap& rMaskBitmap ) SAL_OVERRIDE;
|
|
|
|
|
|
|
|
virtual void drawMask(
|
|
|
|
const SalTwoRect& rPosAry,
|
|
|
|
const SalBitmap& rSalBitmap,
|
|
|
|
SalColor nMaskColor ) SAL_OVERRIDE;
|
|
|
|
|
|
|
|
virtual SalBitmap* getBitmap( long nX, long nY, long nWidth, long nHeight ) SAL_OVERRIDE;
|
|
|
|
|
|
|
|
virtual SalColor getPixel( long nX, long nY ) SAL_OVERRIDE;
|
|
|
|
|
|
|
|
// invert --> ClipRegion (only Windows or VirDevs)
|
|
|
|
virtual void invert(
|
|
|
|
long nX, long nY,
|
|
|
|
long nWidth, long nHeight,
|
|
|
|
SalInvert nFlags) SAL_OVERRIDE;
|
|
|
|
|
|
|
|
virtual void invert( sal_uInt32 nPoints, const SalPoint* pPtAry, SalInvert nFlags ) SAL_OVERRIDE;
|
|
|
|
|
|
|
|
virtual bool drawEPS(
|
|
|
|
long nX, long nY,
|
|
|
|
long nWidth, long nHeight,
|
|
|
|
void* pPtr,
|
|
|
|
sal_uLong nSize ) SAL_OVERRIDE;
|
|
|
|
|
|
|
|
/** Render bitmap with alpha channel
|
|
|
|
|
|
|
|
@param rSourceBitmap
|
|
|
|
Source bitmap to blit
|
|
|
|
|
|
|
|
@param rAlphaBitmap
|
|
|
|
Alpha channel to use for blitting
|
|
|
|
|
|
|
|
@return true, if the operation succeeded, and false
|
|
|
|
otherwise. In this case, clients should try to emulate alpha
|
|
|
|
compositing themselves
|
|
|
|
*/
|
|
|
|
virtual bool drawAlphaBitmap(
|
|
|
|
const SalTwoRect&,
|
|
|
|
const SalBitmap& rSourceBitmap,
|
|
|
|
const SalBitmap& rAlphaBitmap ) SAL_OVERRIDE;
|
|
|
|
|
|
|
|
/** draw transformed bitmap (maybe with alpha) where Null, X, Y define the coordinate system */
|
|
|
|
virtual bool drawTransformedBitmap(
|
|
|
|
const basegfx::B2DPoint& rNull,
|
|
|
|
const basegfx::B2DPoint& rX,
|
|
|
|
const basegfx::B2DPoint& rY,
|
|
|
|
const SalBitmap& rSourceBitmap,
|
|
|
|
const SalBitmap* pAlphaBitmap) SAL_OVERRIDE;
|
|
|
|
|
|
|
|
/** Render solid rectangle with given transparency
|
|
|
|
|
|
|
|
@param nTransparency
|
|
|
|
Transparency value (0-255) to use. 0 blits and opaque, 255 a
|
|
|
|
fully transparent rectangle
|
|
|
|
*/
|
|
|
|
virtual bool drawAlphaRect(
|
|
|
|
long nX, long nY,
|
|
|
|
long nWidth, long nHeight,
|
|
|
|
sal_uInt8 nTransparency ) SAL_OVERRIDE;
|
2014-10-29 17:25:55 +01:00
|
|
|
|
|
|
|
virtual bool drawGradient(const tools::PolyPolygon& rPolygon, const Gradient& rGradient) SAL_OVERRIDE;
|
2014-11-04 17:29:49 -05:00
|
|
|
|
2014-11-12 12:45:36 -05:00
|
|
|
virtual void beginPaint() SAL_OVERRIDE;
|
|
|
|
virtual void endPaint() SAL_OVERRIDE;
|
2014-10-23 23:23:26 +02:00
|
|
|
private:
|
|
|
|
};
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* vim:set shiftwidth=4 softtabstop=4 expandtab: */
|