VirtualBox

source: vbox/trunk/src/VBox/Runtime/r3/solaris/systemmem-solaris.cpp@ 46646

Last change on this file since 46646 was 43880, checked in by vboxsync, 12 years ago

Implemented RTSystemQueryTotalRam and RTSystemQueryAvailableRam on solaris.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 5.8 KB
Line 
1/* $Id: systemmem-solaris.cpp 43880 2012-11-15 14:49:25Z vboxsync $ */
2/** @file
3 * IPRT - RTSystemQueryTotalRam, Solaris ring-3.
4 */
5
6/*
7 * Copyright (C) 2012 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 *
17 * The contents of this file may alternatively be used under the terms
18 * of the Common Development and Distribution License Version 1.0
19 * (CDDL) only, as it comes in the "COPYING.CDDL" file of the
20 * VirtualBox OSE distribution, in which case the provisions of the
21 * CDDL are applicable instead of those of the GPL.
22 *
23 * You may elect to license modified versions of this file under the
24 * terms and conditions of either the GPL or the CDDL or both.
25 */
26
27
28/*******************************************************************************
29* Header Files *
30*******************************************************************************/
31#include <iprt/system.h>
32#include "internal/iprt.h"
33
34#include <iprt/err.h>
35#include <iprt/assert.h>
36#include <iprt/critsect.h>
37#include <iprt/once.h>
38#include <iprt/param.h>
39
40#include <errno.h>
41#include <kstat.h>
42
43
44/*******************************************************************************
45* Global Variables *
46*******************************************************************************/
47/** Initialize globals once. */
48static RTONCE g_rtSysMemSolInitOnce = RTONCE_INITIALIZER;
49/** Critical section serializing access to g_pKStatCtl and the other handles. */
50static RTCRITSECT g_rtSysMemSolCritSect;
51/** The kstate control handle. */
52static kstat_ctl_t *g_pKStatCtl = NULL;
53/** The unix.system_pages handle. */
54static kstat_t *g_pUnixSysPages = NULL;
55/** The zfs.archstats handle. */
56static kstat_t *g_pZfsArcStats = NULL;
57
58
59/** @callback_method_impl{FNRTONCE} */
60static DECLCALLBACK(int) rtSysMemSolInit(void *pvUser)
61{
62 int rc = RTCritSectInit(&g_rtSysMemSolCritSect);
63 if (RT_SUCCESS(rc))
64 {
65 g_pKStatCtl = kstat_open();
66 if (g_pKStatCtl)
67 {
68 g_pUnixSysPages = kstat_lookup(g_pKStatCtl, (char *)"unix", 0, (char *)"system_pages");
69 if (g_pUnixSysPages)
70 {
71 g_pZfsArcStats = kstat_lookup(g_pKStatCtl, (char *)"zfs", 0, (char *)"arcstats"); /* allow NULL */
72 return VINF_SUCCESS;
73 }
74
75 rc = RTErrConvertFromErrno(errno);
76 kstat_close(g_pKStatCtl);
77 g_pKStatCtl = NULL;
78 }
79 else
80 rc = RTErrConvertFromErrno(errno);
81 }
82 return rc;
83}
84
85
86/** @callback_method_impl{FNRTONCECLEANUP} */
87static DECLCALLBACK(void) rtSysMemSolCleanUp(void *pvUser, bool fLazyCleanUpOk)
88{
89 RTCritSectDelete(&g_rtSysMemSolCritSect);
90
91 kstat_close(g_pKStatCtl);
92 g_pKStatCtl = NULL;
93 g_pUnixSysPages = NULL;
94 g_pZfsArcStats = NULL;
95
96 NOREF(pvUser); NOREF(fLazyCleanUpOk);
97}
98
99
100
101RTDECL(int) RTSystemQueryTotalRam(uint64_t *pcb)
102{
103 AssertPtrReturn(pcb, VERR_INVALID_POINTER);
104
105 int rc = RTOnceEx(&g_rtSysMemSolInitOnce, rtSysMemSolInit, rtSysMemSolCleanUp, NULL);
106 if (RT_SUCCESS(rc))
107 {
108 rc = RTCritSectEnter(&g_rtSysMemSolCritSect);
109 if (RT_SUCCESS(rc))
110 {
111 if (kstat_read(g_pKStatCtl, g_pUnixSysPages, NULL) != -1)
112 {
113 kstat_named_t *pData = (kstat_named_t *)kstat_data_lookup(g_pUnixSysPages, (char *)"physmem");
114 if (pData)
115 *pcb = (uint64_t)pData->value.ul * PAGE_SIZE;
116 else
117 rc = RTErrConvertFromErrno(errno);
118 }
119 else
120 rc = RTErrConvertFromErrno(errno);
121 RTCritSectLeave(&g_rtSysMemSolCritSect);
122 }
123 }
124 return rc;
125}
126
127
128RTDECL(int) RTSystemQueryAvailableRam(uint64_t *pcb)
129{
130 AssertPtrReturn(pcb, VERR_INVALID_POINTER);
131
132 int rc = RTOnceEx(&g_rtSysMemSolInitOnce, rtSysMemSolInit, rtSysMemSolCleanUp, NULL);
133 if (RT_SUCCESS(rc))
134 {
135 rc = RTCritSectEnter(&g_rtSysMemSolCritSect);
136 if (RT_SUCCESS(rc))
137 {
138 if (kstat_read(g_pKStatCtl, g_pUnixSysPages, NULL) != -1)
139 {
140 kstat_named_t *pData = (kstat_named_t *)kstat_data_lookup(g_pUnixSysPages, (char *)"freemem");
141 if (pData)
142 {
143 *pcb = (uint64_t)pData->value.ul * PAGE_SIZE;
144
145 /*
146 * Adjust for ZFS greedyness if possible.
147 * (c_min is the target minimum size of the cache, it is not
148 * an absolute minimum.)
149 */
150 if ( g_pZfsArcStats
151 && kstat_read(g_pKStatCtl, g_pZfsArcStats, NULL) != -1)
152 {
153 kstat_named_t *pCurSize = (kstat_named_t *)kstat_data_lookup(g_pZfsArcStats, (char *)"size");
154 kstat_named_t *pMinSize = (kstat_named_t *)kstat_data_lookup(g_pZfsArcStats, (char *)"c_min");
155 if ( pCurSize
156 && pMinSize
157 && pCurSize->value.ul > pMinSize->value.ul)
158 {
159 *pcb += pCurSize->value.ul - pMinSize->value.ul;
160 }
161 }
162 }
163 else
164 rc = RTErrConvertFromErrno(errno);
165 }
166
167 RTCritSectLeave(&g_rtSysMemSolCritSect);
168 }
169 }
170 return rc;
171}
172
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette