Face culling - LearnOpenGL

文章推薦指數: 80 %
投票人數:10人

face culling does. OpenGL checks all the faces that are · front facing towards the viewer and renders those while discarding all the faces that are · back facing ... Ifyou'rerunningAdBlock,pleaseconsiderwhitelistingthissiteifyou'dliketosupportLearnOpenGL;andnoworries,Iwon'tbemadifyoudon't:) IntroductionGettingstartedOpenGLCreatingawindowHelloWindowHelloTriangleShadersTexturesTransformationsCoordinateSystemsCameraReviewLightingColorsBasicLightingMaterialsLightingmapsLightcastersMultiplelightsReviewModelLoadingAssimpMeshModelAdvancedOpenGLDepthtestingStenciltestingBlendingFacecullingFramebuffersCubemapsAdvancedDataAdvancedGLSLGeometryShaderInstancingAntiAliasingAdvancedLightingAdvancedLightingGammaCorrectionShadowsShadowMappingPointShadowsNormalMappingParallaxMappingHDRBloomDeferredShadingSSAOPBRTheoryLightingIBLDiffuseirradianceSpecularIBLInPracticeDebuggingTextRendering2DGameBreakoutSettingupRenderingSpritesLevelsCollisionsBallCollisiondetectionCollisionresolutionParticlesPostprocessingPowerupsAudioRendertextFinalthoughtsGuestArticlesHowtopublish2020OITIntroductionWeightedBlendedSkeletalAnimation2021CSMSceneSceneGraphFrustumCullingTessellationHeightmapTessellationDSACoderepositoryTranslationsAbout BTC 1CLGKgmBSuYJ1nnvDGAepVTKNNDpUjfpRa ETH/ERC20 0x1de59bd9e52521a46309474f8372531533bd7c43 Faceculling Advanced-OpenGL/Face-culling Trymentallyvisualizinga3Dcubeandcountthemaximumnumberoffacesyou'llbeabletoseefromanydirection.Ifyourimaginationisnottoocreativeyouprobablyendedupwithamaximumnumberof3.Youcanviewacubefromanypositionand/ordirection,butyouwouldneverbeabletoseemorethan3faces.Sowhywouldwewastetheeffortofdrawingthoseother3facesthatwecan'tevensee.Ifwecoulddiscardthoseinsomewaywewouldsavemorethan50%ofthiscube'stotalfragmentshaderruns! Wesaymorethan50%insteadof50%,becausefromcertainanglesonly2oreven1facecouldbevisible.Inthatcasewe'dsavemorethan50%. Thisisareallygreatidea,butthere'soneproblemweneedtosolve:howdoweknowifafaceofanobjectisnotvisiblefromtheviewer'spointofview?Ifweimagineanyclosedshape,eachofitsfaceshastwosides.Eachsidewouldeitherfacetheuserorshowitsbacktotheuser.Whatifwecouldonlyrenderthefacesthatarefacingtheviewer? Thisisexactlywhatfacecullingdoes.OpenGLchecksallthefacesthatarefrontfacingtowardstheviewerandrendersthosewhilediscardingallthefacesthatarebackfacing,savingusalotoffragmentshadercalls.WedoneedtotellOpenGLwhichofthefacesweuseareactuallythefrontfacesandwhichfacesarethebackfaces.OpenGLusesaclevertrickforthisbyanalyzingthewindingorderofthevertexdata. Windingorder Whenwedefineasetoftriangleverticeswe'redefiningtheminacertainwindingorderthatiseitherclockwiseorcounter-clockwise.Eachtriangleconsistsof3verticesandwespecifythose3verticesinawindingorderasseenfromthecenterofthetriangle. Asyoucanseeintheimagewefirstdefinevertex1andfromtherewecanchoosewhetherthenextvertexis2or3.Thischoicedefinesthewindingorderofthistriangle.Thefollowingcodeillustratesthis: floatvertices[]={ //clockwise vertices[0],//vertex1 vertices[1],//vertex2 vertices[2],//vertex3 //counter-clockwise vertices[0],//vertex1 vertices[2],//vertex3 vertices[1]//vertex2 }; Eachsetof3verticesthatformatriangleprimitivethuscontainawindingorder.OpenGLusesthisinformationwhenrenderingyourprimitivestodetermineifatriangleisafront-facingoraback-facingtriangle.Bydefault,trianglesdefinedwithcounter-clockwiseverticesareprocessedasfront-facingtriangles. Whendefiningyourvertexorderyouvisualizethecorrespondingtriangleasifitwasfacingyou,soeachtrianglethatyou'respecifyingshouldbecounter-clockwiseasifyou'redirectlyfacingthattriangle.Thecoolthingaboutspecifyingallyourverticeslikethisisthattheactualwindingorderiscalculatedattherasterizationstage,sowhenthevertexshaderhasalreadyrun.Theverticesarethenseenasfromtheviewer'spointofview. Allthetriangleverticesthatthevieweristhenfacingareindeedinthecorrectwindingorderaswespecifiedthem,buttheverticesofthetrianglesattheothersideofthecubearenowrenderedinsuchawaythattheirwindingorderbecomesreversed.Theresultisthatthetriangleswe'refacingareseenasfront-facingtrianglesandthetrianglesatthebackareseenasback-facingtriangles.Thefollowingimageshowsthiseffect: Inthevertexdatawedefinedbothtrianglesincounter-clockwiseorder(thefrontandbacktriangleas1,2,3).However,fromtheviewer'sdirectionthebacktriangleisrenderedclockwiseifwedrawitintheorderof1,2and3fromtheviewer'scurrentpointofview.Eventhoughwespecifiedthebacktriangleincounter-clockwiseorder,itisnowrenderedinaclockwiseorder.Thisisexactlywhatwewanttocull(discard)non-visiblefaces! Faceculling AtthestartofthechapterwesaidthatOpenGLisabletodiscardtriangleprimitivesifthey'rerenderedasback-facingtriangles.NowthatweknowhowtosetthewindingorderoftheverticeswecanstartusingOpenGL'sfacecullingoptionwhichisdisabledbydefault. Thecubevertexdataweusedinthepreviouschapterswasn'tdefinedwiththecounter-clockwisewindingorderinmind,soIupdatedthevertexdatatoreflectacounter-clockwisewindingorderwhichyoucancopyfromhere.It'sagoodpracticetotryandvisualizethattheseverticesareindeedalldefinedinacounter-clockwiseorderforeachtriangle. ToenablefacecullingweonlyhavetoenableOpenGL'sGL_CULL_FACEoption: glEnable(GL_CULL_FACE); Fromthispointon,allthefacesthatarenotfront-facesarediscarded(tryflyinginsidethecubetoseethatallinnerfacesareindeeddiscarded).Currentlywesaveover50%ofperformanceonrenderingfragmentsifOpenGLdecidestorenderthebackfacesfirst(otherwisedepthtestingwould'vediscardedthemalready).Donotethatthisonlyreallyworkswithclosedshapeslikeacube.Wedohavetodisablefacecullingagainwhenwedrawthegrassleavesfromthepreviouschapter,sincetheirfrontandbackfaceshouldbevisible. OpenGLallowsustochangethetypeoffacewewanttocullaswell.Whatifwewanttocullfrontfacesandnotthebackfaces?WecandefinethisbehaviorwithglCullFace: glCullFace(GL_FRONT); TheglCullFacefunctionhasthreepossibleoptions: GL_BACK:Cullsonlythebackfaces. GL_FRONT:Cullsonlythefrontfaces. GL_FRONT_AND_BACK:Cullsboththefrontandbackfaces. TheinitialvalueofglCullFaceisGL_BACK.WecanalsotellOpenGLwe'dratherpreferclockwisefacesasthefront-facesinsteadofcounter-clockwisefacesviaglFrontFace: glFrontFace(GL_CCW); ThedefaultvalueisGL_CCWthatstandsforcounter-clockwiseorderingwiththeotheroptionbeingGL_CWwhich(obviously)standsforclockwiseordering. AsasimpletestwecouldreversethewindingorderbytellingOpenGLthatthefront-facesarenowdeterminedbyaclockwiseorderinginsteadofacounter-clockwiseordering: glEnable(GL_CULL_FACE); glCullFace(GL_BACK); glFrontFace(GL_CW); Theresultisthatonlythebackfacesarerendered: Notethatyoucancreatethesameeffectbycullingfrontfaceswiththedefaultcounter-clockwisewindingorder: glEnable(GL_CULL_FACE); glCullFace(GL_FRONT); Asyoucansee,facecullingisagreattoolforincreasingperformanceofyourOpenGLapplicationswithminimaleffort;especiallyasall3Dapplicationsexportmodelswithconsistentwindingorders(CCWbydefault).Youdohavetokeeptrackoftheobjectsthatwillactuallybenefitfromfacecullingandwhichobjectsshouldn'tbeculledatall. Exercises Canyoure-definethevertexdatabyspecifyingeachtriangleinclockwiseorderandthenrenderthescenewithclockwisetrianglessetasthefrontfaces:solution HI



請為這篇文章評分?