| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246 | 
							- -- 2011-05-24 Pedro Lopes (Microsoft) [email protected] (http://aka.ms/sqlinsights)
 
- --
 
- -- 2012-03-25 Added SQL 2012 support
 
- -- 2012-09-19 Simplified logic
 
- -- 2012-09-20 Changed grow settings if not SQL Server 2012
 
- --
 
- -- Generates the sql statements to preemtively fix VLF issues in all DBs within the server, based on the transaction log current size.
 
- --
 
- SET NOCOUNT ON;
 
- DECLARE @query VARCHAR(1000), @dbname VARCHAR(255), @count int, @usedlogsize bigint, @logsize bigint
 
- DECLARE @sqlcmd NVARCHAR(1000), @sqlparam NVARCHAR(100), @filename VARCHAR(255), @i int, @recmodel NVARCHAR(128)
 
- DECLARE @potsize int, @n_iter int, @n_iter_final int, @initgrow int, @n_init_iter int, @bckpath NVARCHAR(255)
 
- DECLARE @majorver smallint, @minorver smallint, @build smallint
 
- CREATE TABLE #loginfo (dbname varchar(100), num_of_rows int, used_logsize_MB DECIMAL(20,1))
 
- DECLARE @tblvlf TABLE (dbname varchar(100), 
 
- 	Actual_log_size_MB DECIMAL(20,1), 
 
- 	Potential_log_size_MB DECIMAL(20,1), 
 
- 	Actual_VLFs int, 
 
- 	Potential_VLFs int, 
 
- 	Growth_iterations int,
 
- 	Log_Initial_size_MB DECIMAL(20,1), 
 
- 	File_autogrow_MB DECIMAL(20,1))
 
- 	
 
- SELECT TOP 1 @bckpath = REVERSE(RIGHT(REVERSE(physical_device_name), LEN(physical_device_name)-CHARINDEX('\',REVERSE(physical_device_name),0))) FROM msdb.dbo.backupmediafamily WHERE device_type = 2
 
- SELECT @majorver = (@@microsoftversion / 0x1000000) & 0xff, @minorver = (@@microsoftversion / 0x10000) & 0xff, @build = @@microsoftversion & 0xffff
 
-  
 
- --DECLARE csr CURSOR FAST_FORWARD FOR SELECT name FROM master..sysdatabases WHERE dbid > 4 AND DATABASEPROPERTYEX(name,'status') = 'ONLINE' AND DATABASEPROPERTYEX(name,'Updateability') = 'READ_WRITE' AND name <> 'tempdb' AND name <> 'ReportServerTempDB'
 
- DECLARE csr CURSOR FAST_FORWARD FOR SELECT name FROM master.sys.databases WHERE is_read_only = 0 AND state = 0 AND database_id <> 2;
 
- OPEN csr
 
- FETCH NEXT FROM csr INTO @dbname
 
- WHILE (@@FETCH_STATUS <> -1)
 
- BEGIN
 
- 	CREATE TABLE #log_info (recoveryunitid int NULL,
 
- 	fileid tinyint,
 
- 	file_size bigint,
 
- 	start_offset bigint,
 
- 	FSeqNo int,
 
- 	[status] tinyint,
 
- 	parity tinyint,
 
- 	create_lsn numeric(25,0))
 
- 	SET @query = 'DBCC LOGINFO (' + '''' + @dbname + ''') WITH NO_INFOMSGS'
 
- 	IF @majorver < 11
 
- 	BEGIN
 
- 		INSERT INTO #log_info (fileid, file_size, start_offset, FSeqNo, [status], parity, create_lsn)
 
- 		EXEC (@query)
 
- 	END
 
- 	ELSE
 
- 	BEGIN
 
- 		INSERT INTO #log_info (recoveryunitid, fileid, file_size, start_offset, FSeqNo, [status], parity, create_lsn)
 
- 		EXEC (@query)
 
- 	END
 
- 	SET @count = @@ROWCOUNT
 
- 	SET @usedlogsize = (SELECT (MIN(l.start_offset) + SUM(CASE WHEN l.status <> 0 THEN l.file_size ELSE 0 END))/1024.00/1024.00 FROM #log_info l)
 
- 	DROP TABLE #log_info;
 
- 	INSERT #loginfo
 
- 	VALUES(@dbname, @count, @usedlogsize);
 
- 	FETCH NEXT FROM csr INTO @dbname
 
- END
 
- CLOSE csr
 
- DEALLOCATE csr
 
- PRINT '/* Generated on ' + CONVERT (VARCHAR, GETDATE()) + ' in ' + @@SERVERNAME + ' */' + CHAR(10)
 
- 	
 
- DECLARE cshrk CURSOR FAST_FORWARD FOR SELECT dbname, num_of_rows FROM #loginfo 
 
- WHERE num_of_rows >= 50 --My rule of thumb is 50 VLFs. Your mileage may vary.
 
- ORDER BY dbname
 
- OPEN cshrk
 
- FETCH NEXT FROM cshrk INTO @dbname, @count
 
- WHILE (@@FETCH_STATUS <> -1)
 
- BEGIN
 
- 	SET @sqlcmd = 'SELECT @nameout = name, @logsizeout = (size*8)/1024 FROM [' + @dbname + '].dbo.sysfiles WHERE (64 & status) = 64'
 
- 	SET @sqlparam = '@nameout NVARCHAR(100) OUTPUT, @logsizeout bigint OUTPUT'
 
- 	EXEC sp_executesql @sqlcmd, @sqlparam, @nameout = @filename OUTPUT, @logsizeout = @logsize OUTPUT;
 
- 	PRINT '---------------------------------------------------------------------------------------------------------- '
 
- 	PRINT CHAR(13) + 'USE ' + QUOTENAME(@dbname) + ';'
 
- 	PRINT 'DBCC SHRINKFILE (N''' + @filename + ''', 1, TRUNCATEONLY);'
 
- 	PRINT '--'
 
- 	PRINT '-- CHECK: if the tlog file has shrunk with the following query:'
 
- 	PRINT 'SELECT name, (size*8)/1024 AS log_MB FROM [' + @dbname + '].dbo.sysfiles WHERE (64 & status) = 64'
 
- 	PRINT '--'
 
- 	SET @recmodel = CONVERT(NVARCHAR, DATABASEPROPERTYEX(@dbname,'Recovery'))
 
- 	IF @recmodel <> 'SIMPLE'
 
- 	BEGIN
 
- 		PRINT '-- If the log has not shrunk, you must backup the transaction log next.'
 
- 		PRINT '-- Repeat the backup and shrink process alternatively until you get the desired log size (about 1MB).'
 
- 		PRINT '--'
 
- 		PRINT '-- METHOD: Backup -> Shrink (repeat the backup and shrink process until the log has shrunk):'
 
- 		PRINT '--'
 
- 		PRINT '-- Create example logical backup device.' 
 
- 		PRINT 'USE master;' + CHAR(13) + 'EXEC sp_addumpdevice ''disk'', ''BckLog'', ''' + @bckpath + '\example_bck.trn'';'
 
- 		PRINT 'USE ' + QUOTENAME(@dbname) + ';'
 
- 		PRINT '-- Backup Log'
 
- 		PRINT 'BACKUP LOG ' + QUOTENAME(@dbname) + ' TO BckLog;'
 
- 		PRINT '-- Shrink'
 
- 		PRINT 'DBCC SHRINKFILE (N''' + @filename + ''', 1);'
 
- 		PRINT '--'
 
- 		PRINT '-- METHOD: Alter recovery model -> Shrink:'
 
- 		PRINT '-- NOTE: Because the database is in ' + @recmodel + ' recovery model, one alternative is to set it to SIMPLE to truncate the log, shrink it, and reset it to ' + @recmodel + '.'
 
- 		PRINT '-- NOTE2: This method of setting the recovery model to SIMPLE and back again WILL BREAK log chaining, and thus any log shipping or mirroring.'
 
- 		PRINT 'USE [master]; ' + CHAR(13) + 'ALTER DATABASE ' + QUOTENAME(@dbname) + ' SET RECOVERY SIMPLE;'
 
- 		PRINT 'USE ' + QUOTENAME(@dbname) + ';' + CHAR(13) + 'DBCC SHRINKFILE (N''' + @filename + ''', 1);'
 
- 		PRINT 'USE [master]; ' + CHAR(13) + 'ALTER DATABASE ' + QUOTENAME(@dbname) + ' SET RECOVERY ' + @recmodel + ';'
 
- 		PRINT '--'
 
- 		PRINT '-- CHECK: if the tlog file has shrunk with the following query:'
 
- 		PRINT 'SELECT name, (size*8)/1024 AS log_MB FROM [' + @dbname + '].dbo.sysfiles WHERE (64 & status) = 64'
 
- 	END
 
- 	ELSE
 
- 	BEGIN
 
- 		PRINT '-- If not, then proceed to the next step (it may be necessary to execute multiple times):'
 
- 		PRINT 'DBCC SHRINKFILE (N''' + @filename + ''', 1);'
 
- 		PRINT '-- CHECK: if the tlog file has shrunk with the following query:'
 
- 		PRINT 'SELECT name, (size*8)/1024 AS log_MB FROM [' + @dbname + '].dbo.sysfiles WHERE (64 & status) = 64'
 
- 	END
 
- 	-- We are growing in MB instead of GB because of known issue prior to SQL 2012.
 
- 	-- More detail here: http://www.sqlskills.com/BLOGS/PAUL/post/Bug-log-file-growth-broken-for-multiples-of-4GB.aspx
 
- 	-- and http://connect.microsoft.com/SQLServer/feedback/details/481594/log-growth-not-working-properly-with-specific-growth-sizes-vlfs-also-not-created-appropriately
 
- 	-- or https://connect.microsoft.com/SQLServer/feedback/details/357502/transaction-log-file-size-will-not-grow-exactly-4gb-when-filegrowth-4gb
 
- 	IF @majorver >= 11
 
- 	BEGIN
 
- 		SET @n_iter = (SELECT CASE WHEN @logsize <= 64 THEN 1
 
- 			WHEN @logsize > 64 AND @logsize < 256 THEN ROUND(CONVERT(FLOAT, ROUND(@logsize, -2))/256, 0)
 
- 			WHEN @logsize >= 256 AND @logsize < 1024 THEN ROUND(CONVERT(FLOAT, ROUND(@logsize, -2))/512, 0)
 
- 			WHEN @logsize >= 1024 AND @logsize < 4096 THEN ROUND(CONVERT(FLOAT, ROUND(@logsize, -2))/1024, 0)
 
- 			WHEN @logsize >= 4096 AND @logsize < 8192 THEN ROUND(CONVERT(FLOAT, ROUND(@logsize, -2))/2048, 0)
 
- 			WHEN @logsize >= 8192 AND @logsize < 16384 THEN ROUND(CONVERT(FLOAT, ROUND(@logsize, -2))/4096, 0)
 
- 			WHEN @logsize >= 16384 THEN ROUND(CONVERT(FLOAT, ROUND(@logsize, -2))/8192, 0)
 
- 			END)
 
- 		SET @potsize = (SELECT CASE WHEN @logsize <= 64 THEN 1*64
 
- 			WHEN @logsize > 64 AND @logsize < 256 THEN ROUND(CONVERT(FLOAT, ROUND(@logsize, -2))/256, 0)*256
 
- 			WHEN @logsize >= 256 AND @logsize < 1024 THEN ROUND(CONVERT(FLOAT, ROUND(@logsize, -2))/512, 0)*512
 
- 			WHEN @logsize >= 1024 AND @logsize < 4096 THEN ROUND(CONVERT(FLOAT, ROUND(@logsize, -2))/1024, 0)*1024
 
- 			WHEN @logsize >= 4096 AND @logsize < 8192 THEN ROUND(CONVERT(FLOAT, ROUND(@logsize, -2))/2048, 0)*2048
 
- 			WHEN @logsize >= 8192 AND @logsize < 16384 THEN ROUND(CONVERT(FLOAT, ROUND(@logsize, -2))/4096, 0)*4096
 
- 			WHEN @logsize >= 16384 THEN ROUND(CONVERT(FLOAT, ROUND(@logsize, -2))/8192, 0)*8192
 
- 			END)
 
- 	END
 
- 	ELSE
 
- 	BEGIN
 
- 		SET @n_iter = (SELECT CASE WHEN @logsize <= 64 THEN 1
 
- 			WHEN @logsize > 64 AND @logsize < 256 THEN ROUND(CONVERT(FLOAT, ROUND(@logsize, -2))/256, 0)
 
- 			WHEN @logsize >= 256 AND @logsize < 1024 THEN ROUND(CONVERT(FLOAT, ROUND(@logsize, -2))/512, 0)
 
- 			WHEN @logsize >= 1024 AND @logsize < 4096 THEN ROUND(CONVERT(FLOAT, ROUND(@logsize, -2))/1024, 0)
 
- 			WHEN @logsize >= 4096 AND @logsize < 8192 THEN ROUND(CONVERT(FLOAT, ROUND(@logsize, -2))/2048, 0)
 
- 			WHEN @logsize >= 8192 AND @logsize < 16384 THEN ROUND(CONVERT(FLOAT, ROUND(@logsize, -2))/4000, 0)
 
- 			WHEN @logsize >= 16384 THEN ROUND(CONVERT(FLOAT, ROUND(@logsize, -2))/8000, 0)
 
- 			END)
 
- 		SET @potsize = (SELECT CASE WHEN @logsize <= 64 THEN 1*64
 
- 			WHEN @logsize > 64 AND @logsize < 256 THEN ROUND(CONVERT(FLOAT, ROUND(@logsize, -2))/256, 0)*256
 
- 			WHEN @logsize >= 256 AND @logsize < 1024 THEN ROUND(CONVERT(FLOAT, ROUND(@logsize, -2))/512, 0)*512
 
- 			WHEN @logsize >= 1024 AND @logsize < 4096 THEN ROUND(CONVERT(FLOAT, ROUND(@logsize, -2))/1024, 0)*1024
 
- 			WHEN @logsize >= 4096 AND @logsize < 8192 THEN ROUND(CONVERT(FLOAT, ROUND(@logsize, -2))/2048, 0)*2048
 
- 			WHEN @logsize >= 8192 AND @logsize < 16384 THEN ROUND(CONVERT(FLOAT, ROUND(@logsize, -2))/4000, 0)*4000
 
- 			WHEN @logsize >= 16384 THEN ROUND(CONVERT(FLOAT, ROUND(@logsize, -2))/8000, 0)*8000
 
- 			END)
 
- 	END
 
- 	
 
- 	-- If the proposed log size is smaller than current log, and also smaller than 4GB,
 
- 	-- and there is less than 512MB of diff between the current size and proposed size, add 1 grow.
 
- 	SET @n_iter_final = @n_iter
 
- 	IF @logsize > @potsize AND @potsize <= 4096 AND ABS(@logsize - @potsize) < 512
 
- 	BEGIN
 
- 		SET @n_iter_final = @n_iter + 1
 
- 	END
 
- 	-- If the proposed log size is larger than current log, and also larger than 50GB, 
 
- 	-- and there is less than 1GB of diff between the current size and proposed size, take 1 grow.
 
- 	ELSE IF @logsize < @potsize AND @potsize <= 51200 AND ABS(@logsize - @potsize) > 1024
 
- 	BEGIN
 
- 		SET @n_iter_final = @n_iter - 1
 
- 	END
 
- 	IF @potsize = 0 
 
- 	BEGIN 
 
- 		SET @potsize = 64 
 
- 	END
 
- 	IF @n_iter = 0 
 
- 	BEGIN 
 
- 		SET @n_iter = 1
 
- 	END
 
- 	
 
- 	SET @potsize = (SELECT CASE WHEN @n_iter < @n_iter_final THEN @potsize + (@potsize/@n_iter) 
 
- 			WHEN @n_iter > @n_iter_final THEN @potsize - (@potsize/@n_iter) 
 
- 			ELSE @potsize END)
 
- 	
 
- 	SET @n_init_iter = @n_iter_final
 
- 	IF @potsize >= 8192
 
- 	BEGIN
 
- 		SET @initgrow = @potsize/@n_iter_final
 
- 	END
 
- 	IF @potsize >= 64 AND @potsize <= 512
 
- 	BEGIN
 
- 		SET @n_init_iter = 1
 
- 		SET @initgrow = 512
 
- 	END
 
- 	IF @potsize > 512 AND @potsize <= 1024
 
- 	BEGIN
 
- 		SET @n_init_iter = 1
 
- 		SET @initgrow = 1023
 
- 	END
 
- 	IF @potsize > 1024 AND @potsize < 8192
 
- 	BEGIN
 
- 		SET @n_init_iter = 1
 
- 		SET @initgrow = @potsize
 
- 	END
 
- 	INSERT INTO @tblvlf
 
- 	SELECT @dbname, @logsize, @potsize, @count, 
 
- 		CASE WHEN @potsize <= 64 THEN (@potsize/(@potsize/@n_init_iter))*4
 
- 			WHEN @potsize > 64 AND @potsize < 1024 THEN (@potsize/(@potsize/@n_init_iter))*8
 
- 			WHEN @potsize >= 1024 THEN (@potsize/(@potsize/@n_init_iter))*16
 
- 			END, 
 
- 		@n_init_iter, @initgrow, CASE WHEN (@potsize/@n_iter_final) <= 1024 THEN (@potsize/@n_iter_final) ELSE 1024 END
 
- 	
 
- 	SET @i = 0
 
- 	WHILE @i <= @n_init_iter
 
- 	BEGIN
 
- 		IF @i = 1
 
- 		BEGIN
 
- 			--Log Autogrow should not be above 1GB
 
- 			PRINT CHAR(13) + '-- Now for the log file growth:'
 
- 			PRINT 'ALTER DATABASE [' + @dbname + '] MODIFY FILE ( NAME = N''' + @filename + ''', SIZE = ' + CONVERT(VARCHAR, @initgrow) + 'MB , FILEGROWTH = ' + CASE WHEN (@potsize/@n_iter_final) <= 1024 THEN CONVERT(VARCHAR, (@potsize/@n_iter_final)) ELSE '1024' END + 'MB );'
 
- 		END
 
- 		IF @i > 1
 
- 		BEGIN
 
- 			PRINT 'ALTER DATABASE [' + @dbname + '] MODIFY FILE ( NAME = N''' + @filename + ''', SIZE = ' + CONVERT(VARCHAR, @initgrow*@i)+ 'MB );'
 
- 		END		
 
- 		SET @i = @i + 1
 
- 		CONTINUE
 
- 	END
 
- 	FETCH NEXT FROM cshrk INTO @dbname, @count
 
- END
 
- CLOSE cshrk
 
- DEALLOCATE cshrk;
 
- DROP TABLE #loginfo;
 
- SELECT dbname AS [Database_Name], Actual_log_size_MB, Potential_log_size_MB, Actual_VLFs, 
 
- 	Potential_VLFs, Growth_iterations, Log_Initial_size_MB, File_autogrow_MB
 
- FROM @tblvlf;
 
- GO
 
 
  |